| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471547254735474547554765477547854795480548154825483548454855486548754885489549054915492549354945495549654975498549955005501550255035504550555065507550855095510551155125513551455155516551755185519552055215522552355245525552655275528552955305531553255335534553555365537553855395540554155425543554455455546554755485549555055515552555355545555555655575558555955605561556255635564556555665567556855695570557155725573557455755576557755785579558055815582558355845585558655875588558955905591559255935594559555965597559855995600560156025603560456055606560756085609561056115612561356145615561656175618561956205621562256235624562556265627562856295630563156325633563456355636563756385639564056415642564356445645564656475648564956505651565256535654565556565657565856595660566156625663566456655666566756685669567056715672567356745675567656775678567956805681568256835684568556865687568856895690569156925693 | //---------------------------------------------------------------------------#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 "S3FileSystem.h"#include "FileMasks.h"#include <Soap.EncdDecd.hpp>#include <StrUtils.hpp>#include <XMLDoc.hpp>#include <StrUtils.hpp>#include <algorithm>//---------------------------------------------------------------------------#pragma package(smart_init)//---------------------------------------------------------------------------#define SET_SESSION_PROPERTY_FROM(PROPERTY, FROM) \  if (F##PROPERTY != FROM) { F##PROPERTY = FROM; Modify(); }//---------------------------------------------------------------------------#define SET_SESSION_PROPERTY(PROPERTY) \  SET_SESSION_PROPERTY_FROM(PROPERTY, value)//---------------------------------------------------------------------------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 UnicodeString HostKeyNames[HOSTKEY_COUNT] = {L"WARN", L"rsa", L"dsa", L"ecdsa", L"ed25519", L"ed448"};const UnicodeString GssLibNames[GSSLIB_COUNT] = {L"gssapi32", L"sspi", L"custom"};// Update also order in Ssh2CipherList()const TCipher DefaultCipherList[CIPHER_COUNT] =  { cipAES, cipChaCha20, cipBlowfish, cip3DES, cipWarn, cipArcfour, cipDES };// Update also order in SshKexList()const TKex DefaultKexList[KEX_COUNT] =  { kexECDH, kexDHGEx, kexDHGroup14, kexRSA, kexWarn, kexDHGroup1 };const THostKey DefaultHostKeyList[HOSTKEY_COUNT] =  { hkED448, hkED25519, hkECDSA, hkRSA, hkDSA, hkWarn };const TGssLib DefaultGssLibList[GSSLIB_COUNT] =  { gssGssApi32, gssSspi, gssCustom };const wchar_t FSProtocolNames[FSPROTOCOL_COUNT][16] = { L"SCP", L"SFTP (SCP)", L"SFTP", L"", L"", L"FTP", L"WebDAV", L"S3" };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 WebDAVProtocol(L"dav");const UnicodeString WebDAVSProtocol(L"davs");const UnicodeString S3Protocol(L"s3");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 UrlRawSettingsParamNamePrefix(L"x-");const UnicodeString PassphraseOption(L"passphrase");const UnicodeString RawSettingsOption(L"rawsettings");const UnicodeString S3HostName(S3LibDefaultHostName());const UnicodeString OpensshHostDirective(L"Host");//---------------------------------------------------------------------TDateTime __fastcall SecToDateTime(int Sec){  return TDateTime(double(Sec) / SecsPerDay);}//---------------------------------------------------------------------static bool IsValidOpensshLine(const UnicodeString & Line){  return !Line.IsEmpty() && (Line[1] != L'#');}//---------------------------------------------------------------------static bool ParseOpensshDirective(const UnicodeString & ALine, UnicodeString & Directive, UnicodeString & Value){  bool Result = IsValidOpensshLine(ALine);  if (Result)  {    int From = 1;    wchar_t Equal = L'=';    UnicodeString Delimiters(UnicodeString(L" \t") + Equal);    wchar_t Delimiter;    UnicodeString Line = Trim(ALine);    Directive = CopyToChars(Line, From, Delimiters, false, &Delimiter, false);    Result = !Directive.IsEmpty();    if (Result)    {      Value = Line;      Value.Delete(1, Directive.Length() + 1);      Value = Trim(Value);      if ((Delimiter != Equal) && !Value.IsEmpty() && (Value[1] == Equal))      {        Value.Delete(1, 1);        Value = Trim(Value); // not sure about this, but for the directives we support it does not matter      }      Result = !Value.IsEmpty();    }  }  return Result;}//--- 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::DefaultSettings(){  HostName = L"";  PortNumber = SshPortNumber;  UserName = L"";  Password = L"";  NewPassword = L"";  ChangePassword = false;  PingInterval = 30;  PingType = ptOff;  Timeout = 15;  TryAgent = true;  AgentFwd = false;  AuthKI = true;  AuthKIPassword = true;  AuthGSSAPI = true;  AuthGSSAPIKEX = false;  GSSAPIFwdTGT = false;  LogicalHostName = L"";  ChangeUsername = false;  Compression = false;  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];  }  for (int Index = 0; Index < HOSTKEY_COUNT; Index++)  {    HostKeys[Index] = DefaultHostKeyList[Index];  }  for (int Index = 0; Index < GSSLIB_COUNT; Index++)  {    GssLib[Index] = DefaultGssLibList[Index];  }  GssLibCustom = L"";  PublicKeyFile = L"";  Passphrase = L"";  FPuttyProtocol = L"";  TcpNoDelay = false;  SendBuf = DefaultSendBuf;  SourceAddress = L"";  ProtocolFeatures = L"";  SshSimple = true;  HostKey = L"";  FingerprintScan = false;  FOverrideCachedHostKey = true;  Note = L"";  WinTitle = L"";  InternalEditorEncoding = -1;  EncryptKey = UnicodeString();  WebDavLiberalEscaping = false;  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;  ResolveSymlinks = true;  FollowDirectorySymlinks = false;  DSTMode = dstmUnix;  DeleteToRecycleBin = false;  OverwrittenToRecycleBin = false;  RecycleBinPath = L"";  Color = 0;  PostLoginCommands = L"";  // SCP  LookupUserGroups = asAuto;  EOLType = eolLF;  TrimVMSVersions = false;  VMSAllRevisions = false;  Shell = L""; //default shell  ReturnVar = L"";  ExitCode1IsError = false;  ClearAliases = true;  UnsetNationalVars = true;  ListingCommand = L"ls -la";  IgnoreLsWarnings = true;  Scp1Compatibility = false;  TimeDifference = 0;  TimeDifferenceAuto = true;  SCPLsFullTime = asAuto;  NotUtf = asAuto;  // S3  S3DefaultRegion = L"";  S3SessionToken = L"";  S3UrlStyle = s3usVirtualHost;  S3MaxKeys = asAuto;  S3CredentialsEnv = false;  // SFTP  SftpServer = L"";  SFTPDownloadQueue = 32;  SFTPUploadQueue = 32;  SFTPListingQueue = 2;  SFTPMaxVersion = ::SFTPMaxVersion;  SFTPMaxPacketSize = 0;  SFTPRealPath = asAuto;  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"";  TunnelPassphrase = 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 = tls13;  FtpListAll = asAuto;  FtpHost = asAuto;  FtpWorkFromCwd = asAuto;  FtpAnyCodeForPwd = false;  SslSessionReuse = true;  TlsCertificateFile = L"";  FtpProxyLogonType = 0; // none  PuttySettings = UnicodeString();  CustomParam1 = L"";  CustomParam2 = L"";}//---------------------------------------------------------------------void __fastcall TSessionData::Default(){  DefaultSettings();  IsWorkspace = false;  Link = L"";  NameOverride = L"";  Selected = false;  FModified = false;  FSource = ::ssNone;  FSaveOnly = false;  // add also to TSessionLog::AddStartupInfo()}//---------------------------------------------------------------------void __fastcall TSessionData::NonPersistant(){  UpdateDirectories = false;  PreserveDirectoryChanges = false;}//---------------------------------------------------------------------#define PROPERTY(P) PROPERTY_HANDLER(P, )#define BASE_PROPERTIES \  PROPERTY(HostName); \  PROPERTY(PortNumber); \  PROPERTY(UserName); \  PROPERTY_HANDLER(Password, F); \  PROPERTY(PublicKeyFile); \  PROPERTY_HANDLER(Passphrase, F); \  PROPERTY(FSProtocol); \  PROPERTY(Ftps); \  PROPERTY(LocalDirectory); \  PROPERTY(RemoteDirectory); \  PROPERTY(Color); \  PROPERTY(SynchronizeBrowsing); \  PROPERTY(Note);//---------------------------------------------------------------------#define ADVANCED_PROPERTIES \  PROPERTY_HANDLER(NewPassword, F); \  PROPERTY(ChangePassword); \  PROPERTY(PingInterval); \  PROPERTY(PingType); \  PROPERTY(Timeout); \  PROPERTY(TryAgent); \  PROPERTY(AgentFwd); \  PROPERTY(LogicalHostName); \  PROPERTY(ChangeUsername); \  PROPERTY(Compression); \  PROPERTY(Ssh2DES); \  PROPERTY(SshNoUserAuth); \  PROPERTY(CipherList); \  PROPERTY(KexList); \  PROPERTY(HostKeyList); \  PROPERTY(GssLibList); \  PROPERTY(GssLibCustom); \  PROPERTY(AddressFamily); \  PROPERTY(RekeyData); \  PROPERTY(RekeyTime); \  PROPERTY(HostKey); \  PROPERTY(FingerprintScan); \  PROPERTY(InternalEditorEncoding); \  \  PROPERTY(UpdateDirectories); \  PROPERTY(CacheDirectories); \  PROPERTY(CacheDirectoryChanges); \  PROPERTY(PreserveDirectoryChanges); \  \  PROPERTY(ResolveSymlinks); \  PROPERTY(FollowDirectorySymlinks); \  PROPERTY(DSTMode); \  PROPERTY(Special); \  PROPERTY(Selected); \  PROPERTY(ReturnVar); \  PROPERTY(ExitCode1IsError); \  PROPERTY(LookupUserGroups); \  PROPERTY(EOLType); \  PROPERTY(TrimVMSVersions); \  PROPERTY(VMSAllRevisions); \  PROPERTY(Shell); \  PROPERTY(ClearAliases); \  PROPERTY(Scp1Compatibility); \  PROPERTY(UnsetNationalVars); \  PROPERTY(ListingCommand); \  PROPERTY(IgnoreLsWarnings); \  PROPERTY(SCPLsFullTime); \  \  PROPERTY(TimeDifference); \  PROPERTY(TimeDifferenceAuto); \  PROPERTY(TcpNoDelay); \  PROPERTY(SendBuf); \  PROPERTY(SourceAddress); \  PROPERTY(ProtocolFeatures); \  PROPERTY(SshSimple); \  PROPERTY(AuthKI); \  PROPERTY(AuthKIPassword); \  PROPERTY(AuthGSSAPI); \  PROPERTY(AuthGSSAPIKEX); \  PROPERTY(GSSAPIFwdTGT); \  PROPERTY(DeleteToRecycleBin); \  PROPERTY(OverwrittenToRecycleBin); \  PROPERTY(RecycleBinPath); \  PROPERTY(NotUtf); \  PROPERTY(PostLoginCommands); \  \  PROPERTY(S3DefaultRegion); \  PROPERTY(S3SessionToken); \  PROPERTY(S3UrlStyle); \  PROPERTY(S3MaxKeys); \  PROPERTY(S3CredentialsEnv); \  \  PROPERTY(ProxyMethod); \  PROPERTY(ProxyHost); \  PROPERTY(ProxyPort); \  PROPERTY(ProxyUsername); \  PROPERTY_HANDLER(ProxyPassword, F); \  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); \  PROPERTY(SFTPRealPath); \  \  for (unsigned int Index = 0; Index < LENOF(FSFTPBugs); Index++) \  { \    PROPERTY(SFTPBug[(TSftpBug)Index]); \  } \  \  PROPERTY(Tunnel); \  PROPERTY(TunnelHostName); \  PROPERTY(TunnelPortNumber); \  PROPERTY(TunnelUserName); \  PROPERTY_HANDLER(TunnelPassword, F); \  PROPERTY(TunnelPublicKeyFile); \  PROPERTY_HANDLER(TunnelPassphrase, F); \  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(FtpWorkFromCwd); \  PROPERTY(FtpAnyCodeForPwd); \  PROPERTY(SslSessionReuse); \  PROPERTY(TlsCertificateFile); \  \  PROPERTY(FtpProxyLogonType); \  \  PROPERTY(MinTlsVersion); \  PROPERTY(MaxTlsVersion); \  \  PROPERTY(WinTitle); \  \  PROPERTY_HANDLER(EncryptKey, F); \  \  PROPERTY(WebDavLiberalEscaping); \  \  PROPERTY(PuttySettings); \  \  PROPERTY(CustomParam1); \  PROPERTY(CustomParam2);#define META_PROPERTIES \  PROPERTY(IsWorkspace); \  PROPERTY(Link); \  PROPERTY(NameOverride);//---------------------------------------------------------------------void __fastcall TSessionData::Assign(TPersistent * Source){  if (Source && Source->InheritsFrom(__classid(TSessionData)))  {    TSessionData * SourceData = (TSessionData *)Source;    // Master password prompt shows implicitly here, when cloning the session data for a new terminal    CopyData(SourceData);    FSource = SourceData->FSource;  }  else  {    TNamedObject::Assign(Source);  }}//---------------------------------------------------------------------void __fastcall TSessionData::DoCopyData(TSessionData * SourceData, bool NoRecrypt){  #define PROPERTY_HANDLER(P, F) \    if (NoRecrypt) \    { \      F##P = SourceData->F##P; \    } \    else \    { \      P = SourceData->P; \    }  PROPERTY(Name);  BASE_PROPERTIES;  ADVANCED_PROPERTIES;  META_PROPERTIES;  #undef PROPERTY_HANDLER  FOverrideCachedHostKey = SourceData->FOverrideCachedHostKey;  FModified = SourceData->Modified;  FSaveOnly = SourceData->FSaveOnly;}//---------------------------------------------------------------------void __fastcall TSessionData::CopyData(TSessionData * SourceData){  DoCopyData(SourceData, false);}//---------------------------------------------------------------------void __fastcall TSessionData::CopyDataNoRecrypt(TSessionData * SourceData){  DoCopyData(SourceData, true);}//---------------------------------------------------------------------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 Decrypted){  bool Result = true;  #define PROPERTY_HANDLER(P, F) \    if ((Decrypted && (P != Default->P)) || \        (!Decrypted && (F##P != Default->F##P))) \    { \      Result = false; \      if (DifferentProperties != NULL) \      { \        DifferentProperties->Add(#P); \      } \      else \      { \        return Result; \      } \    }  if (!AdvancedOnly)  {    BASE_PROPERTIES;    META_PROPERTIES;  }  ADVANCED_PROPERTIES;  #undef PROPERTY_HANDLER  return Result;}//---------------------------------------------------------------------bool __fastcall TSessionData::IsSame(const TSessionData * Default, bool AdvancedOnly){  return IsSame(Default, AdvancedOnly, NULL, false);}//---------------------------------------------------------------------bool __fastcall TSessionData::IsSameDecrypted(const TSessionData * Default){  return IsSame(Default, false, NULL, true);}//---------------------------------------------------------------------TFSProtocol NormalizeFSProtocol(TFSProtocol FSProtocol){  if (FSProtocol == fsSFTPonly)  {    FSProtocol = fsSFTP;  }  return FSProtocol;}//---------------------------------------------------------------------bool __fastcall TSessionData::IsSameSite(const TSessionData * Other){  return    // Particularly when handling /refresh,    // fsSFTPonly sites when compared against sftp:// URLs (fsSFTP) have to match.    // But similarly also falled back SCP sites.    (NormalizeFSProtocol(FSProtocol) == NormalizeFSProtocol(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 PuttyImport, bool & RewritePassword, bool Unsafe){  // 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);  #define LOAD_PASSWORD_EX(PROP, PLAIN_NAME, ENC_NAME, ONPLAIN) \    if (Storage->ValueExists(PLAIN_NAME)) \    { \      PROP = Storage->ReadString(PLAIN_NAME, PROP); \      ONPLAIN \    } \    else \    { \      RawByteString A##PROP = Storage->ReadStringAsBinaryData(ENC_NAME, F##PROP); \      SET_SESSION_PROPERTY_FROM(PROP, A##PROP); \    }  #define LOAD_PASSWORD(PROP, PLAIN_NAME) LOAD_PASSWORD_EX(PROP, PLAIN_NAME, TEXT(#PROP), RewritePassword = true;)  if (!Configuration->DisablePasswordStoring)  {    LOAD_PASSWORD(Password, L"PasswordPlain");  }  HostKey = Storage->ReadString(L"SshHostKey", HostKey); // probably never used  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);  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));  AuthGSSAPIKEX = Storage->ReadBool(L"AuthGSSAPIKEX", AuthGSSAPIKEX);  GSSAPIFwdTGT = Storage->ReadBool(L"GSSAPIFwdTGT", Storage->ReadBool(L"GssapiFwd", Storage->ReadBool(L"SSPIFwdTGT", GSSAPIFwdTGT)));  // KerbPrincipal was used by Quest PuTTY  // GSSAPIServerRealm was used by Vaclav Tomec  LogicalHostName = Storage->ReadString(L"LogicalHostName", Storage->ReadString(L"GSSAPIServerRealm", Storage->ReadString(L"KerbPrincipal", LogicalHostName)));  ChangeUsername = Storage->ReadBool(L"ChangeUsername", ChangeUsername);  Compression = Storage->ReadBool(L"Compression", Compression);  Ssh2DES = Storage->ReadBool(L"Ssh2DES", Ssh2DES);  SshNoUserAuth = Storage->ReadBool(L"SshNoUserAuth", SshNoUserAuth);  CipherList = Storage->ReadString(L"Cipher", CipherList);  KexList = Storage->ReadString(L"KEX", KexList);  HostKeyList = Storage->ReadString(L"HostKey", HostKeyList);  if (!Unsafe)  {    GssLibList = Storage->ReadString(L"GSSLibs", GssLibList);  }  GssLibCustom = Storage->ReadString(L"GSSCustom", GssLibCustom);  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);  Special = Storage->ReadBool(L"Special", Special);  if (!Unsafe)  {    Shell = Storage->ReadString(L"Shell", Shell);  }  ClearAliases = Storage->ReadBool(L"ClearAliases", ClearAliases);  UnsetNationalVars = Storage->ReadBool(L"UnsetNationalVars", UnsetNationalVars);  if (!Unsafe)  {    ListingCommand = Storage->ReadString(L"ListingCommand",      Storage->ReadBool(L"AliasGroupList", false) ? UnicodeString(L"ls -gla") : ListingCommand);  }  IgnoreLsWarnings = Storage->ReadBool(L"IgnoreLsWarnings", IgnoreLsWarnings);  SCPLsFullTime = Storage->ReadEnum(L"SCPLsFullTime", SCPLsFullTime, AutoSwitchMapping);  Scp1Compatibility = Storage->ReadBool(L"Scp1Compatibility", Scp1Compatibility);  TimeDifference = Storage->ReadFloat(L"TimeDifference", TimeDifference);  TimeDifferenceAuto = Storage->ReadBool(L"TimeDifferenceAuto", (TimeDifference == TDateTime()));  if (!Unsafe)  {    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);  }  ExitCode1IsError = Storage->ReadBool(L"ExitCode1IsError", ExitCode1IsError);  LookupUserGroups = Storage->ReadEnum(L"LookupUserGroups2", LookupUserGroups, AutoSwitchMapping);  EOLType = (TEOLType)Storage->ReadInteger(L"EOLType", EOLType);  TrimVMSVersions = Storage->ReadBool(L"TrimVMSVersions", TrimVMSVersions);  VMSAllRevisions = Storage->ReadBool(L"VMSAllRevisions", VMSAllRevisions);  NotUtf = Storage->ReadEnum(L"Utf", Storage->ReadEnum(L"SFTPUtfBug", NotUtf), AutoSwitchReversedMapping);  InternalEditorEncoding = Storage->ReadInteger(L"InternalEditorEncoding", InternalEditorEncoding);  S3DefaultRegion = Storage->ReadString(L"S3DefaultRegion", S3DefaultRegion);  S3SessionToken = Storage->ReadString(L"S3SessionToken", S3SessionToken);  S3UrlStyle = (TS3UrlStyle)Storage->ReadInteger(L"S3UrlStyle", S3UrlStyle);  S3MaxKeys = Storage->ReadEnum(L"S3MaxKeys", S3MaxKeys, AutoSwitchMapping);  S3CredentialsEnv = Storage->ReadBool(L"S3CredentialsEnv", S3CredentialsEnv);  // PuTTY defaults to TcpNoDelay, but the psftp/pscp ignores this preference, and always set this to off (what is our default too)  if (!PuttyImport)  {    TcpNoDelay = Storage->ReadBool(L"TcpNoDelay", TcpNoDelay);  }  SendBuf = Storage->ReadInteger(L"SendBuf", Storage->ReadInteger("SshSendBuf", SendBuf));  SourceAddress = Storage->ReadString(L"SourceAddress", SourceAddress);  ProtocolFeatures = Storage->ReadString(L"ProtocolFeatures", ProtocolFeatures);  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);  // proxy password is not rewritten  LOAD_PASSWORD_EX(ProxyPassword, L"ProxyPassword", L"ProxyPasswordEnc", );  if (!Unsafe)  {    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(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);  READ_BUG(ChanReq);  #undef READ_BUG  if ((Bug[sbHMAC2] == asAuto) &&      Storage->ReadBool(L"BuggyMAC", false))  {      Bug[sbHMAC2] = asOn;  }  if (!Unsafe)  {    SftpServer = Storage->ReadString(L"SftpServer", SftpServer);  }  #define READ_SFTP_BUG(BUG) \    SFTPBug[sb##BUG] = Storage->ReadEnum(L"SFTP" #BUG "Bug", SFTPBug[sb##BUG], AutoSwitchMapping);  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);  SFTPRealPath = Storage->ReadEnum(L"SFTPRealPath", SFTPRealPath, AutoSwitchMapping);  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)  {    LOAD_PASSWORD(TunnelPassword, L"TunnelPasswordPlain");  }  TunnelPublicKeyFile = Storage->ReadString(L"TunnelPublicKeyFile", TunnelPublicKeyFile);  // Contrary to main session passphrase (which has -passphrase switch in scripting),  // we are loading tunnel passphrase, as there's no other way to provide it in scripting  if (!Configuration->DisablePasswordStoring)  {    LOAD_PASSWORD(TunnelPassphrase, L"TunnelPassphrasePlain");  }  TunnelLocalPortNumber = Storage->ReadInteger(L"TunnelLocalPortNumber", TunnelLocalPortNumber);  TunnelHostKey = Storage->ReadString(L"TunnelHostKey", TunnelHostKey);  // Ftp prefix  FtpPasvMode = Storage->ReadBool(L"FtpPasvMode", FtpPasvMode);  FtpForcePasvIp = Storage->ReadEnum(L"FtpForcePasvIp2", FtpForcePasvIp, AutoSwitchMapping);  FtpUseMlsd = Storage->ReadEnum(L"FtpUseMlsd", FtpUseMlsd, AutoSwitchMapping);  FtpAccount = Storage->ReadString(L"FtpAccount", FtpAccount);  FtpPingInterval = Storage->ReadInteger(L"FtpPingInterval", FtpPingInterval);  FtpPingType = static_cast<TPingType>(Storage->ReadInteger(L"FtpPingType", FtpPingType));  FtpTransferActiveImmediately = Storage->ReadEnum(L"FtpTransferActiveImmediately2", FtpTransferActiveImmediately, AutoSwitchMapping);  Ftps = static_cast<TFtps>(Storage->ReadInteger(L"Ftps", Ftps));  FtpListAll = Storage->ReadEnum(L"FtpListAll", FtpListAll, AutoSwitchMapping);  FtpHost = Storage->ReadEnum(L"FtpHost", FtpHost, AutoSwitchMapping);  FtpWorkFromCwd = Storage->ReadEnum(L"FtpWorkFromCwd", Storage->ReadEnum(L"FtpDeleteFromCwd", FtpWorkFromCwd), AutoSwitchMapping);  FtpAnyCodeForPwd = Storage->ReadBool(L"FtpAnyCodeForPwd", FtpAnyCodeForPwd);  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));  LOAD_PASSWORD(EncryptKey, L"EncryptKeyPlain");  WebDavLiberalEscaping = Storage->ReadBool(L"WebDavLiberalEscaping", WebDavLiberalEscaping);  IsWorkspace = Storage->ReadBool(L"IsWorkspace", IsWorkspace);  Link = Storage->ReadString(L"Link", Link);  NameOverride = Storage->ReadString(L"NameOverride", NameOverride);  PuttySettings = Storage->ReadString(L"PuttySettings", PuttySettings);  CustomParam1 = Storage->ReadString(L"CustomParam1", CustomParam1);  CustomParam2 = Storage->ReadString(L"CustomParam2", CustomParam2);  #undef LOAD_PASSWORD#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  #define HOSTKEY_TEST(VALUE, EXPECTED) HostKeyList = VALUE; DebugAssert(HostKeyList == EXPECTED);  #define HOSTKEY_DEFAULT L"ed25519,ecdsa,rsa,dsa,WARN"  // Empty source should result in default list  HOSTKEY_TEST(L"", HOSTKEY_DEFAULT);  // Missing priority algo  HOSTKEY_TEST(L"ecdsa,rsa,dsa,WARN", HOSTKEY_DEFAULT);  // Missing non-priority algo  HOSTKEY_TEST(L"ed25519,ecdsa,dsa,WARN", L"ed25519,ecdsa,dsa,rsa,WARN");  #undef HOSTKEY_DEFAULT  #undef HOSTKEY_TEST#endif}//---------------------------------------------------------------------void __fastcall TSessionData::Load(THierarchicalStorage * Storage, bool PuttyImport){  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, PuttyImport, RewritePassword, false);    Storage->CloseSubKey();  }  if (RewritePassword)  {    TStorageAccessMode AccessMode = Storage->AccessMode;    Storage->AccessMode = smReadWrite;    try    {      if (Storage->OpenSubKey(InternalStorageKey, true))      {        #define REWRITE_PASSWORD(PROP, PLAIN_NAME) \          Storage->DeleteValue(PLAIN_NAME); \          if (!PROP.IsEmpty()) \          { \            Storage->WriteBinaryDataAsString(TEXT(#PROP), F##PROP); \          }        REWRITE_PASSWORD(Password, L"PasswordPlain");        REWRITE_PASSWORD(TunnelPassword, L"TunnelPasswordPlain");        REWRITE_PASSWORD(EncryptKey, L"EncryptKeyPlain");        REWRITE_PASSWORD(TunnelPassphrase, L"TunnelPassphrasePlain");        #undef REWRITE_PASSWORD        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){  // Same as in TCopyParamType::Save  #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, AuthKI);  WRITE_DATA(Bool, AuthKIPassword);  WRITE_DATA_EX(String, L"SshHostKey", HostKey, );  WRITE_DATA(String, Note);  WRITE_DATA(Bool, AuthGSSAPI);  WRITE_DATA(Bool, AuthGSSAPIKEX);  WRITE_DATA(Bool, GSSAPIFwdTGT);  Storage->DeleteValue(L"TryGSSKEX");  Storage->DeleteValue(L"UserNameFromEnvironment");  Storage->DeleteValue("GSSAPIServerChoosesUserName");  Storage->DeleteValue(L"GSSAPITrustDNS");  WRITE_DATA(String, LogicalHostName);  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, );    WRITE_DATA_EX(String, L"KerbPrincipal", LogicalHostName, );    // 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(Bool, Ssh2DES);  WRITE_DATA(Bool, SshNoUserAuth);  WRITE_DATA_EX(String, L"Cipher", CipherList, );  WRITE_DATA_EX(String, L"KEX", KexList, );  WRITE_DATA_EX(String, L"HostKey", HostKeyList, );  WRITE_DATA_EX(String, L"GSSLibs", GssLibList, );  WRITE_DATA_EX(String, L"GSSCustom", GssLibCustom, );  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, );    // 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(Bool, ExitCode1IsError);    WRITE_DATA_EX(Integer, L"LookupUserGroups2", LookupUserGroups, );    WRITE_DATA(Integer, EOLType);    WRITE_DATA(Bool, TrimVMSVersions);    WRITE_DATA(Bool, VMSAllRevisions);    Storage->DeleteValue(L"SFTPUtfBug");    WRITE_DATA_EX(Integer, L"Utf", NotUtf, );    WRITE_DATA(Integer, InternalEditorEncoding);    WRITE_DATA(String, S3DefaultRegion);    WRITE_DATA(String, S3SessionToken);    WRITE_DATA(Integer, S3UrlStyle);    WRITE_DATA(Integer, S3MaxKeys);    WRITE_DATA(Bool, S3CredentialsEnv);    WRITE_DATA(Integer, SendBuf);    WRITE_DATA(String, SourceAddress);    WRITE_DATA(String, ProtocolFeatures);    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(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);  WRITE_BUG(ChanReq);  #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(), );    WRITE_DATA(String, WinTitle);  }  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, SFTPRealPath);    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(String, TunnelHostKey);    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(Integer, FtpWorkFromCwd);    WRITE_DATA(Bool, FtpAnyCodeForPwd);    WRITE_DATA(Bool, SslSessionReuse);    WRITE_DATA(String, TlsCertificateFile);    WRITE_DATA(Integer, FtpProxyLogonType);    WRITE_DATA(Integer, MinTlsVersion);    WRITE_DATA(Integer, MaxTlsVersion);    WRITE_DATA(Bool, WebDavLiberalEscaping);    WRITE_DATA(Bool, IsWorkspace);    WRITE_DATA(String, Link);    WRITE_DATA(String, NameOverride);    WRITE_DATA(String, PuttySettings);    WRITE_DATA(String, CustomParam1);    WRITE_DATA(String, CustomParam2);  }  // This is for collecting all keys for TSiteRawDialog::AddButtonClick.  // It should be enough to test for (Default == NULL),  // the DoNotEncryptPasswords and PuttyExport were added to limit a possible unintended impact.  bool SaveAll = (Default == NULL) && DoNotEncryptPasswords && !PuttyExport;  SavePasswords(Storage, PuttyExport, DoNotEncryptPasswords, SaveAll);  if (PuttyExport)  {    WritePuttySettings(Storage, PuttySettings);  }}//---------------------------------------------------------------------TStrings * __fastcall TSessionData::SaveToOptions(const TSessionData * Default, bool SaveName, bool PuttyExport){  std::unique_ptr<TStringList> Options(new TStringList());  std::unique_ptr<TOptionsStorage> OptionsStorage(new TOptionsStorage(Options.get(), true));  if (SaveName)  {    OptionsStorage->WriteString(L"Name", Name);  }  DoSave(OptionsStorage.get(), PuttyExport, 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;}//---------------------------------------------------------------------_di_IXMLNode __fastcall TSessionData::FindSettingsNode(_di_IXMLNode Node, const UnicodeString & Name){  for (int Index = 0; Index < Node->ChildNodes->Count; Index++)  {    _di_IXMLNode ChildNode = Node->ChildNodes->Get(Index);    if (ChildNode->NodeName == L"Setting")    {       OleVariant SettingName = ChildNode->GetAttribute(L"name");       if (SettingName == Name)       {         return ChildNode;       }    }  }  return NULL;}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::ReadSettingsNode(_di_IXMLNode Node, const UnicodeString & Name, const UnicodeString & Default){  _di_IXMLNode TheNode = FindSettingsNode(Node, Name);  UnicodeString Result;  if (TheNode != NULL)  {    Result = TheNode->Text.Trim();  }  if (Result.IsEmpty())  {    Result = Default;  }  return Result;}//---------------------------------------------------------------------int __fastcall TSessionData::ReadSettingsNode(_di_IXMLNode Node, const UnicodeString & Name, int Default){  _di_IXMLNode TheNode = FindSettingsNode(Node, 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, _di_IXMLNode SettingsNode){  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;    }  }  PublicKeyFile = ReadXmlNode(Node, L"Keyfile", PublicKeyFile);  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;  }  else if (SettingsNode != NULL)  {    int PasvMode = ReadSettingsNode(SettingsNode, L"Use Pasv mode", 1);    FtpPasvMode = (PasvMode != 0);  }  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);  if (SettingsNode != NULL)  {    if (UsesSsh && PublicKeyFile.IsEmpty())    {      UnicodeString KeyFiles = ReadSettingsNode(SettingsNode, L"SFTP keyfiles", UnicodeString());      UnicodeString KeyFile = CutToChar(KeyFiles, L'\n', true).Trim();      KeyFiles = KeyFiles.Trim();      // If there are more keys, ignore them, as we do not know which one to use      if (!KeyFile.IsEmpty() && KeyFiles.IsEmpty())      {        PublicKeyFile = KeyFile;      }    }    bool BypassProxy = (ReadXmlNode(Node, L"BypassProxy", 0) != 0);    if (!BypassProxy)    {      int FtpProxyType = ReadSettingsNode(SettingsNode, L"FTP Proxy type", -1);      if (FtpProxyType > 0)      {        switch (FtpProxyType)        {          case 1:            FtpProxyLogonType = 2;            break;          case 2:            FtpProxyLogonType = 1;            break;          case 3:            FtpProxyLogonType = 3;            break;          case 4:            // custom            // TODO: map known sequences to our enumeration            FtpProxyLogonType = 0;            break;          default:            DebugFail();            FtpProxyLogonType = 0;            break;        }        ProxyHost = ReadSettingsNode(SettingsNode, L"FTP Proxy host", ProxyHost);        ProxyUsername = ReadSettingsNode(SettingsNode, L"FTP Proxy user", ProxyUsername);        ProxyPassword = ReadSettingsNode(SettingsNode, L"FTP Proxy password", ProxyPassword);        // ProxyPort is not used with FtpProxyLogonType      }      else      {        int ProxyType = ReadSettingsNode(SettingsNode, L"Proxy type", -1);        if (ProxyType >= 0)        {          switch (ProxyType)          {            case 0:              ProxyMethod = ::pmNone;              break;            case 1:              ProxyMethod = pmHTTP;              break;            case 2:              ProxyMethod = pmSocks5;              break;            case 3:              ProxyMethod = pmSocks4;              break;            default:              DebugFail();              ProxyMethod = ::pmNone;              break;          }          ProxyHost = ReadSettingsNode(SettingsNode, L"Proxy host", ProxyHost);          ProxyPort = ReadSettingsNode(SettingsNode, L"Proxy port", ProxyPort);          ProxyUsername = ReadSettingsNode(SettingsNode, L"Proxy user", ProxyUsername);          ProxyPassword = ReadSettingsNode(SettingsNode, L"Proxy password", ProxyPassword);        }      }    }  }}//---------------------------------------------------------------------bool OpensshBoolValue(const UnicodeString & Value){  return SameText(Value, L"yes");}//---------------------------------------------------------------------UnicodeString CutOpensshToken(UnicodeString & S){  DebugAssert(!S.IsEmpty() && (S == Trim(S)));  int From = 1;  UnicodeString Result = CopyToChars(S, From, L" \t", false);  S.Delete(1, Result.Length());  S = Trim(S);  DebugAssert(!Result.IsEmpty() && (Result == Trim(Result)));  Result = Trim(Result);  return Result;}//---------------------------------------------------------------------void TSessionData::ImportFromOpenssh(TStrings * Lines){  bool SkippingSection = false;  std::unique_ptr<TStrings> UsedDirectives(CreateSortedStringList());  for (int Index = 0; Index < Lines->Count; Index++)  {    UnicodeString Line = Lines->Strings[Index];    UnicodeString Directive, Value;    if (ParseOpensshDirective(Line, Directive, Value))    {      if (SameText(Directive, OpensshHostDirective))      {        DebugAssert(!Value.IsEmpty() && (Value == Trim(Value)));        SkippingSection = true;        while (!Value.IsEmpty())        {          UnicodeString M = CutOpensshToken(Value);          bool Negated = DebugAlwaysTrue(!M.IsEmpty()) && (M[1] == L'!');          if (Negated)          {            M.Delete(1, 1);          }          TFileMasks Mask;          Mask.SetMask(M);          // This does way more than OpenSSH, but on the other hand, the special characters of our file masks,          // should not be present in hostnames.          if (Mask.Matches(Name, false, UnicodeString(), NULL))          {            if (Negated)            {              // Skip even if matched by other possitive patterns              SkippingSection = true;              break;            }            else            {              // Keep looking, in case if negated              SkippingSection = false;            }          }        }      }      else if (SameText(Directive, L"Match"))      {        SkippingSection = true;      }      else if (!SkippingSection && (UsedDirectives->IndexOf(Directive) < 0))      {        if (SameText(Directive, L"AddressFamily"))        {          if (SameText(Value, L"inet"))          {            AddressFamily = afIPv4;          }          else if (SameText(Value, L"inet6"))          {            AddressFamily = afIPv6;          }          else          {            AddressFamily = afAuto;          }        }        else if (SameText(Directive, L"BindAddress"))        {          SourceAddress = Value;        }        else if (SameText(Directive, L"Compression"))        {          Compression = OpensshBoolValue(Value);        }        else if (SameText(Directive, L"ForwardAgent"))        {          AgentFwd = OpensshBoolValue(Value);        }        else if (SameText(Directive, L"GSSAPIAuthentication"))        {          AuthGSSAPI = OpensshBoolValue(Value);        }        else if (SameText(Directive, L"GSSAPIDelegateCredentials"))        {          AuthGSSAPIKEX = OpensshBoolValue(Value);        }        else if (SameText(Directive, L"Hostname"))        {          HostName = Value;        }        else if (SameText(Directive, L"IdentityFile"))        {          // It's likely there would be forward slashes in OpenSSH config file and our load/save dialogs          // (e.g. when converting keys) work suboptimally when working with forward slashes.          PublicKeyFile = GetNormalizedPath(Value);        }        else if (SameText(Directive, L"KbdInteractiveAuthentication"))        {          AuthKI = OpensshBoolValue(Value);        }        else if (SameText(Directive, L"Port"))        {          PortNumber = StrToInt(Value);        }        else if (SameText(Directive, L"User"))        {          UserName = Value;        }        else if (SameText(Directive, L"ProxyJump"))        {          UnicodeString Jump = Value;          // multiple jumps are not supported          if (Jump.Pos(L",") == 0)          {            std::unique_ptr<TSessionData> JumpData(new TSessionData(EmptyStr));            bool DefaultsOnly;            if ((JumpData->ParseUrl(Jump, NULL, NULL, DefaultsOnly, NULL, NULL, NULL, 0)) &&                !JumpData->HostName.IsEmpty())            {              JumpData->Name = JumpData->HostName;              JumpData->ImportFromOpenssh(Lines);              Tunnel = true;              TunnelHostName = JumpData->HostName;              TunnelPortNumber = JumpData->PortNumber;              TunnelUserName = JumpData->UserName;              TunnelPassword = JumpData->Password;              TunnelPublicKeyFile = JumpData->PublicKeyFile;            }          }        }        UsedDirectives->Add(Directive);      }    }  }}//---------------------------------------------------------------------void __fastcall TSessionData::SavePasswords(THierarchicalStorage * Storage, bool PuttyExport, bool DoNotEncryptPasswords, bool SaveAll){  // It's probably safe to replace this with if (!PuttyExport) { SAVE_PASSWORD(...) }  if (!Configuration->DisablePasswordStoring && !PuttyExport && (!FPassword.IsEmpty() || SaveAll))  {    if (DoNotEncryptPasswords)    {      Storage->WriteString(L"PasswordPlain", Password);      Storage->DeleteValue(L"Password");    }    else    {      Storage->WriteBinaryDataAsString(L"Password", StronglyRecryptPassword(FPassword, UserName+HostName));      Storage->DeleteValue(L"PasswordPlain");    }  }  else  {    Storage->DeleteValue(L"Password");    Storage->DeleteValue(L"PasswordPlain");  }  if (PuttyExport)  {    // save password unencrypted    Storage->WriteString(L"ProxyPassword", ProxyPassword);  }  else  {    #define SAVE_PASSWORD_EX(PROP, PLAIN_NAME, ENC_NAME, ENC_KEY, COND) \      if (DoNotEncryptPasswords) \      { \        if (!F##PROP.IsEmpty() || SaveAll) \        { \          Storage->WriteString(PLAIN_NAME, PROP); \        } \        else \        { \          Storage->DeleteValue(PLAIN_NAME); \        } \        Storage->DeleteValue(ENC_NAME); \      } \      else \      { \        if (COND && (!F##PROP.IsEmpty() || SaveAll)) \        { \          Storage->WriteBinaryDataAsString(ENC_NAME, StronglyRecryptPassword(F##PROP, ENC_KEY)); \        } \        else \        { \          Storage->DeleteValue(ENC_NAME); \        } \        Storage->DeleteValue(PLAIN_NAME); \      }    #define SAVE_PASSWORD(PROP, PLAIN_NAME, ENC_KEY) SAVE_PASSWORD_EX(PROP, PLAIN_NAME, TEXT(#PROP), ENC_KEY, !Configuration->DisablePasswordStoring)    SAVE_PASSWORD_EX(ProxyPassword, L"ProxyPassword", L"ProxyPasswordEnc", ProxyUsername + ProxyHost, true);    SAVE_PASSWORD(TunnelPassword, L"TunnelPasswordPlain", TunnelUserName + TunnelHostName);    SAVE_PASSWORD_EX(EncryptKey, L"EncryptKeyPlain", L"EncryptKey", UserName + HostName, true);  }}//---------------------------------------------------------------------void __fastcall TSessionData::RecryptPasswords(){  Password = Password;  NewPassword = NewPassword;  ProxyPassword = ProxyPassword;  TunnelPassword = TunnelPassword;  TunnelPassphrase = TunnelPassphrase;  Passphrase = Passphrase;  EncryptKey = EncryptKey;}//---------------------------------------------------------------------bool __fastcall TSessionData::HasPassword(){  return !FPassword.IsEmpty();}//---------------------------------------------------------------------bool __fastcall TSessionData::HasAnySessionPassword(){  // Keep in sync with ClearSessionPasswords  return    HasPassword() ||    !FTunnelPassword.IsEmpty() ||    // will probably be never used    !FNewPassword.IsEmpty();}//---------------------------------------------------------------------bool __fastcall TSessionData::HasAnyPassword(){  // Keep in sync with MaskPasswords  return    HasAnySessionPassword() ||    !FProxyPassword.IsEmpty() ||    !FEncryptKey.IsEmpty() ||    !FPassphrase.IsEmpty() ||    !FTunnelPassphrase.IsEmpty();}//---------------------------------------------------------------------void __fastcall TSessionData::ClearSessionPasswords(){  // Keep in sync with HasAnySessionPassword  FPassword = L"";  FNewPassword = 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, false);    }    __finally    {      Storage->CloseSubKey();    }  }}//---------------------------------------------------------------------void __fastcall TSessionData::Remove(THierarchicalStorage * Storage, const UnicodeString & Name){  Storage->RecursiveDeleteSubKey(Name);}//---------------------------------------------------------------------void __fastcall TSessionData::Remove(){  bool SessionList = true;  THierarchicalStorage * Storage = Configuration->CreateScpStorage(SessionList);  try  {    Storage->Explicit = true;    if (Storage->OpenSubKey(Configuration->StoredSessionsSubKey, false))    {      Remove(Storage, InternalStorageKey);    }  }  __finally  {    delete Storage;  }}//---------------------------------------------------------------------void __fastcall TSessionData::CacheHostKeyIfNotCached(){  UnicodeString KeyType = KeyTypeFromFingerprint(HostKey);  // Should allow importing to INI file as ImportHostKeys  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 a checksum of a 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) ||    SameText(Option, PASSWORD_SWITCH) ||    SameText(Option, NEWPASSWORD_SWITCH);}//---------------------------------------------------------------------bool __fastcall TSessionData::IsOptionWithParameters(const UnicodeString & Option){  return SameText(Option, RawSettingsOption);}//---------------------------------------------------------------------bool __fastcall TSessionData::MaskPasswordInOptionParameter(const UnicodeString & Option, UnicodeString & Param){  bool Result = false;  if (SameText(Option, RawSettingsOption))  {    int P = Param.Pos(L"=");    if (P > 0)    {      // TStrings.IndexOfName does not trim      UnicodeString Key = Param.SubString(1, P - 1);      if (SameText(Key, L"ProxyPassword") ||          SameText(Key, L"ProxyPasswordEnc") ||          SameText(Key, L"TunnelPassword") ||          SameText(Key, L"TunnelPasswordPlain") ||          SameText(Key, L"TunnelPassphrase") ||          SameText(Key, L"TunnelPassphrasePlain") ||          SameText(Key, L"EncryptKey") ||          SameText(Key, L"EncryptKeyPlain"))      {        Param = Key + L"=" + PasswordMask;        Result = true;      }    }  }  return Result;}//---------------------------------------------------------------------void __fastcall TSessionData::MaskPasswords(){  // Keep in sync with HasAnyPassword  if (!Password.IsEmpty())  {    Password = PasswordMask;  }  if (!NewPassword.IsEmpty())  {    NewPassword = PasswordMask;  }  if (!ProxyPassword.IsEmpty())  {    ProxyPassword = PasswordMask;  }  if (!TunnelPassword.IsEmpty())  {    TunnelPassword = PasswordMask;  }  if (!TunnelPassphrase.IsEmpty())  {    TunnelPassphrase = PasswordMask;  }  if (!EncryptKey.IsEmpty())  {    EncryptKey = PasswordMask;  }  if (!Passphrase.IsEmpty())  {    Passphrase = PasswordMask;  }}//---------------------------------------------------------------------static bool IsDomainOrSubdomain(const UnicodeString & FullDomain, const UnicodeString & Domain){  return    SameText(FullDomain, Domain) ||    EndsText(L"." + Domain, FullDomain);}//---------------------------------------------------------------------bool __fastcall TSessionData::ParseUrl(UnicodeString Url, TOptions * Options,  TStoredSessionList * StoredSessions, bool & DefaultsOnly, UnicodeString * FileName,  bool * AProtocolDefined, UnicodeString * MaskedUrl, int Flags){  bool ProtocolDefined = false;  bool PortNumberDefined = false;  TFSProtocol AFSProtocol;  int DefaultProtocolPortNumber;  TFtps AFtps = ftpsNone;  int ProtocolLen = 0;  if (IsProtocolUrl(Url, ScpProtocol, ProtocolLen))  {    AFSProtocol = fsSCPonly;    DefaultProtocolPortNumber = SshPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  else if (IsProtocolUrl(Url, SftpProtocol, ProtocolLen))  {    AFSProtocol = fsSFTPonly;    DefaultProtocolPortNumber = SshPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  else if (IsProtocolUrl(Url, FtpProtocol, ProtocolLen))  {    AFSProtocol = fsFTP;    Ftps = ftpsNone;    DefaultProtocolPortNumber = FtpPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  else if (IsProtocolUrl(Url, FtpsProtocol, ProtocolLen))  {    AFSProtocol = fsFTP;    AFtps = ftpsImplicit;    DefaultProtocolPortNumber = FtpsImplicitPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  else if (IsProtocolUrl(Url, FtpesProtocol, ProtocolLen))  {    AFSProtocol = fsFTP;    AFtps = ftpsExplicitTls;    DefaultProtocolPortNumber = FtpPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  else if (IsProtocolUrl(Url, WebDAVProtocol, ProtocolLen) ||           IsProtocolUrl(Url, HttpProtocol, ProtocolLen))  {    AFSProtocol = fsWebDAV;    AFtps = ftpsNone;    DefaultProtocolPortNumber = HTTPPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  else if (IsProtocolUrl(Url, WebDAVSProtocol, ProtocolLen) ||           IsProtocolUrl(Url, HttpsProtocol, ProtocolLen))  {    AFSProtocol = fsWebDAV;    AFtps = ftpsImplicit;    DefaultProtocolPortNumber = HTTPSPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  else if (IsProtocolUrl(Url, S3Protocol, ProtocolLen))  {    AFSProtocol = fsS3;    AFtps = ftpsImplicit;    DefaultProtocolPortNumber = 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;    DefaultProtocolPortNumber = SshPortNumber;    MoveStr(Url, MaskedUrl, ProtocolLen);    ProtocolDefined = true;  }  if (ProtocolDefined && (Url.SubString(1, 2) == L"//"))  {    MoveStr(Url, MaskedUrl, 2);  }  if (AProtocolDefined != NULL)  {    *AProtocolDefined = ProtocolDefined;  }  bool Unsafe = FLAGSET(Flags, pufUnsafe);  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) &&        (!ProtocolDefined || FLAGSET(Flags, pufAllowStoredSiteWithProtocol)))    {      // 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())      {        int APortNumber = StrToIntDef(DecodeUrlChars(HostInfo), -1);        if ((APortNumber > 0) && (APortNumber <= 65535))        {          PortNumber = APortNumber;          PortNumberDefined = true;        }      }      else if (ProtocolDefined)      {        if ((AFSProtocol == fsWebDAV) &&            (IsDomainOrSubdomain(HostName, S3LibDefaultHostName()) ||             IsDomainOrSubdomain(HostName, L"digitaloceanspaces.com") ||             IsDomainOrSubdomain(HostName, L"storage.googleapis.com")))        {          AFSProtocol = fsS3;        }        PortNumber = DefaultProtocolPortNumber;      }      if (ProtocolDefined)      {        Ftps = AFtps;      }      UnicodeString UserInfoWithoutConnectionParams = CutToChar(UserInfo, UrlParamSeparator, false);      UnicodeString ConnectionParams = UserInfo;      UserInfo = UserInfoWithoutConnectionParams;      std::unique_ptr<TStrings> RawSettings(new TStringList());      while (!ConnectionParams.IsEmpty())      {        UnicodeString ConnectionParam = CutToChar(ConnectionParams, UrlParamSeparator, false);        UnicodeString ConnectionParamName = CutToChar(ConnectionParam, UrlParamValueSeparator, false);        if (SameText(ConnectionParamName, UrlHostKeyParamName))        {          HostKey = DecodeUrlChars(ConnectionParam);          FOverrideCachedHostKey = false;        }        else if (StartsText(UrlRawSettingsParamNamePrefix, ConnectionParamName))        {          UnicodeString AName = RightStr(ConnectionParamName, ConnectionParamName.Length() - UrlRawSettingsParamNamePrefix.Length());          AName = DecodeUrlChars(AName);          UnicodeString Value = DecodeUrlChars(ConnectionParam);          if (SameText(AName, L"Name"))          {            Name = Value;          }          else          {            RawSettings->Values[AName] = Value;          }        }      }      if (RawSettings->Count > 0) // optimization      {        ApplyRawSettings(RawSettings.get(), FLAGSET(Flags, pufUnsafe));      }      bool HasPassword = (UserInfo.Pos(L':') > 0);      UnicodeString RawUserName = CutToChar(UserInfo, L':', false);      UserName = DecodeUrlChars(RawUserName);      Password = DecodeUrlChars(UserInfo);      if (HasPassword && Password.IsEmpty())      {        Password = EmptyString;      }      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 (HasPassword)        {          (*MaskedUrl) += L":" + PasswordMask;        }        if (!RawUserName.IsEmpty() || HasPassword)        {          (*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(USERNAME_SWITCH, Value))    {      UserName = Value;    }    if (Options->FindSwitch(PASSWORD_SWITCH, Value))    {      Password = Value;    }    if (Options->FindSwitch(SESSIONNAME_SWICH, Value))    {      Name = Value;    }    if (Options->FindSwitch(NEWPASSWORD_SWITCH, Value))    {      ChangePassword = true;      NewPassword = 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(RawSettingsOption))    {      std::unique_ptr<TStrings> RawSettings(new TStringList());      if (Options->FindSwitch(RawSettingsOption, RawSettings.get()))      {        ApplyRawSettings(RawSettings.get(), Unsafe);      }    }  }  return true;}//---------------------------------------------------------------------void __fastcall TSessionData::ApplyRawSettings(TStrings * RawSettings, bool Unsafe){  std::unique_ptr<TOptionsStorage> OptionsStorage(new TOptionsStorage(RawSettings, false));  ApplyRawSettings(OptionsStorage.get(), Unsafe);}//---------------------------------------------------------------------void __fastcall TSessionData::ApplyRawSettings(THierarchicalStorage * Storage, bool Unsafe){  bool Dummy;  DoLoad(Storage, false, Dummy, Unsafe);}//---------------------------------------------------------------------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"";}//---------------------------------------------------------------------TSessionData * TSessionData::CreateTunnelData(int TunnelLocalPortNumber){  std::unique_ptr<TSessionData> TunnelData(new TSessionData(EmptyStr));  TunnelData->Assign(StoredSessions->DefaultSettings);  TunnelData->Name = FMTLOAD(TUNNEL_SESSION_NAME, (SessionName));  TunnelData->Tunnel = false;  TunnelData->HostName = TunnelHostName;  TunnelData->PortNumber = TunnelPortNumber;  TunnelData->UserName = TunnelUserName;  TunnelData->Password = TunnelPassword;  TunnelData->PublicKeyFile = TunnelPublicKeyFile;  TunnelData->Passphrase = TunnelPassphrase;  UnicodeString AHostName = HostNameExpanded;  if (IsIPv6Literal(AHostName))  {    AHostName = EscapeIPv6Literal(AHostName);  }  TunnelData->TunnelPortFwd = FORMAT(L"L%d\t%s:%d",    (TunnelLocalPortNumber, AHostName, PortNumber));  TunnelData->HostKey = TunnelHostKey;  // inherit proxy options on the main session  TunnelData->ProxyMethod = ProxyMethod;  TunnelData->ProxyHost = ProxyHost;  TunnelData->ProxyPort = ProxyPort;  TunnelData->ProxyUsername = ProxyUsername;  TunnelData->ProxyPassword = ProxyPassword;  TunnelData->ProxyTelnetCommand = ProxyTelnetCommand;  TunnelData->ProxyLocalCommand = ProxyLocalCommand;  TunnelData->ProxyDNS = ProxyDNS;  TunnelData->ProxyLocalhost = ProxyLocalhost;  // inherit most SSH options of the main session (except for private key and bugs)  TunnelData->Compression = Compression;  TunnelData->CipherList = CipherList;  TunnelData->Ssh2DES = Ssh2DES;  TunnelData->KexList = KexList;  TunnelData->RekeyData = RekeyData;  TunnelData->RekeyTime = RekeyTime;  TunnelData->SshNoUserAuth = SshNoUserAuth;  TunnelData->AuthGSSAPI = AuthGSSAPI;  TunnelData->AuthGSSAPIKEX = AuthGSSAPIKEX;  TunnelData->GSSAPIFwdTGT = GSSAPIFwdTGT;  TunnelData->TryAgent = TryAgent;  TunnelData->AgentFwd = AgentFwd;  TunnelData->AuthKI = AuthKI;  TunnelData->AuthKIPassword = AuthKIPassword;  return TunnelData.release();}//---------------------------------------------------------------------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();}//---------------------------------------------------------------------------int TSessionData::GetDefaultPort(){  return DefaultPort(FSProtocol, Ftps);}//---------------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetSessionKey(){  UnicodeString Result = FORMAT(L"%s@%s", (UserName, HostName));  if (PortNumber != GetDefaultPort())  {    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;    UnicodeString XNewPassword = NewPassword;    UnicodeString XEncryptKey = EncryptKey;    // 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;    NewPassword = XNewPassword;    EncryptKey = XEncryptKey;    Shred(XPassword);    Shred(XNewPassword);    Shred(XEncryptKey);  }}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetHostNameExpanded(){  return ::ExpandEnvironmentVariables(HostName);}//---------------------------------------------------------------------UnicodeString TSessionData::GetHostNameSource(){  UnicodeString Result;  if (HostName != HostNameExpanded)  {    Result = HostName;  }  return Result;}//---------------------------------------------------------------------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;    UnicodeString XNewPassword = NewPassword;    UnicodeString XEncryptKey = EncryptKey;    SET_SESSION_PROPERTY(UserName);    Password = XPassword;    NewPassword = XNewPassword;    EncryptKey = XEncryptKey;    Shred(XPassword);    Shred(XNewPassword);    Shred(XEncryptKey);  }}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetUserNameExpanded(){  UnicodeString Result = ::ExpandEnvironmentVariables(UserName);  if (Result.IsEmpty() && HasS3AutoCredentials())  {    Result = S3EnvUserName();  }  return Result;}//---------------------------------------------------------------------UnicodeString TSessionData::GetUserNameSource(){  UnicodeString Result;  if (UserName.IsEmpty() && HasS3AutoCredentials())  {    S3EnvUserName(&Result);  }  if (Result.IsEmpty() && (UserName != UserNameExpanded))  {    Result = UserName;  }  return Result;}//---------------------------------------------------------------------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::SetNewPassword(UnicodeString avalue){  RawByteString value = EncryptPassword(avalue, UserName+HostName);  SET_SESSION_PROPERTY(NewPassword);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetNewPassword() const{  return DecryptPassword(FNewPassword, UserName+HostName);}//---------------------------------------------------------------------void __fastcall TSessionData::SetChangePassword(bool value){  SET_SESSION_PROPERTY(ChangePassword);}//---------------------------------------------------------------------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::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::SetAuthGSSAPIKEX(bool value){  SET_SESSION_PROPERTY(AuthGSSAPIKEX);}//---------------------------------------------------------------------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::SetSsh2DES(bool value){  SET_SESSION_PROPERTY(Ssh2DES);}//---------------------------------------------------------------------void __fastcall TSessionData::SetSshNoUserAuth(bool value){  SET_SESSION_PROPERTY(SshNoUserAuth);}//---------------------------------------------------------------------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);  bool HasWarnAlgo = (WarnAlgo >= AlgoT());  const AlgoT * WarnPtr;  int WarnDefaultIndex;  if (!HasWarnAlgo)  {    WarnPtr = NULL;    WarnDefaultIndex = -1;  }  else  {    WarnPtr = std::find(DefaultList, DefaultList + Count, WarnAlgo);    DebugAssert(WarnPtr != NULL);    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 (HasWarnAlgo && !Used[WarnAlgo] && DebugAlwaysTrue(Index < Count))  {    NewList[Index] = WarnAlgo;    Used[WarnAlgo] = true;    Index++;  }  int WarnIndex = -1;  if (HasWarnAlgo)  {    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 (HasWarnAlgo && (DefaultIndex < WarnDefaultIndex))        {          TargetIndex = WarnIndex;        }        else        {          TargetIndex = Index;        }      }      NewList.insert(NewList.begin() + TargetIndex, DefaultAlgo);      DebugAssert(NewList.back() == AlgoT());      NewList.pop_back();      if (HasWarnAlgo && (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::SetHostKeys(int Index, THostKey value){  DebugAssert(Index >= 0 && Index < HOSTKEY_COUNT);  SET_SESSION_PROPERTY(HostKeys[Index]);}//---------------------------------------------------------------------THostKey __fastcall TSessionData::GetHostKeys(int Index) const{  DebugAssert(Index >= 0 && Index < HOSTKEY_COUNT);  return FHostKeys[Index];}//---------------------------------------------------------------------void __fastcall TSessionData::SetHostKeyList(UnicodeString value){  SetAlgoList(FHostKeys, DefaultHostKeyList, HostKeyNames, HOSTKEY_COUNT, hkWarn, value);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetHostKeyList() const{  UnicodeString Result;  for (int Index = 0; Index < HOSTKEY_COUNT; Index++)  {    Result += UnicodeString(Index ? L"," : L"") + HostKeyNames[HostKeys[Index]];  }  return Result;}//---------------------------------------------------------------------void __fastcall TSessionData::SetGssLib(int Index, TGssLib value){  DebugAssert(Index >= 0 && Index < GSSLIB_COUNT);  SET_SESSION_PROPERTY(GssLib[Index]);}//---------------------------------------------------------------------TGssLib __fastcall TSessionData::GetGssLib(int Index) const{  DebugAssert(Index >= 0 && Index < GSSLIB_COUNT);  return FGssLib[Index];}//---------------------------------------------------------------------void __fastcall TSessionData::SetGssLibList(UnicodeString value){  SetAlgoList(FGssLib, DefaultGssLibList, GssLibNames, GSSLIB_COUNT, TGssLib(-1), value);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetGssLibList() const{  UnicodeString Result;  for (int Index = 0; Index < GSSLIB_COUNT; Index++)  {    Result += UnicodeString(Index ? L"," : L"") + GssLibNames[GssLib[Index]];  }  return Result;}//---------------------------------------------------------------------void __fastcall TSessionData::SetGssLibCustom(UnicodeString value){  SET_SESSION_PROPERTY(GssLibCustom);}//---------------------------------------------------------------------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::SetExitCode1IsError(bool value){  SET_SESSION_PROPERTY(ExitCode1IsError);}//---------------------------------------------------------------------------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);}//---------------------------------------------------------------------------void __fastcall TSessionData::SetVMSAllRevisions(bool value){  SET_SESSION_PROPERTY(VMSAllRevisions);}//---------------------------------------------------------------------------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(){  UnicodeString Result;  if (!HostName.IsEmpty() && !UserName.IsEmpty())  {    // If we ever choose to include port number,    // we have to escape IPv6 literals in HostName    Result = FORMAT(L"%s@%s", (UserName, HostName));  }  else if (!HostName.IsEmpty())  {    Result = HostName;  }  else  {    Result = L"session";  }  Result = MakeValidName(Result);  return Result;}//---------------------------------------------------------------------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));}//---------------------------------------------------------------------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:    case fsS3:      Result = (Ftps != ftpsNone);      break;    default:      DebugFail();      break;  }  return Result;}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetProtocolUrl(bool HttpForWebDAV){  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 (HttpForWebDAV)      {        if (Ftps == ftpsImplicit)        {          Url = HttpsProtocol;        }        else        {          Url = HttpProtocol;        }      }      else      {        if (Ftps == ftpsImplicit)        {          Url = WebDAVSProtocol;        }        else        {          Url = WebDAVProtocol;        }      }      break;    case fsS3:      Url = S3Protocol;      break;  }  Url += ProtocolSeparator;  return Url;}//---------------------------------------------------------------------static bool HasIP6LiteralBrackets(const UnicodeString & HostName){  return    (HostName.Length() >= 2) &&    (HostName[1] == L'[') &&    (HostName[HostName.Length()] == L']');}//---------------------------------------------------------------------static UnicodeString StripIP6LiteralBrackets(const UnicodeString & HostName){  UnicodeString Result = HostName;  if (DebugAlwaysTrue(HasIP6LiteralBrackets(Result)))  {    Result = Result.SubString(2, Result.Length() - 2);  }  return Result;}//---------------------------------------------------------------------bool __fastcall IsIPv6Literal(const UnicodeString & HostName){  UnicodeString Buf = HostName;  if (HasIP6LiteralBrackets(Buf))  {    Buf = StripIP6LiteralBrackets(Buf);  }  int Colons = 0;  bool Result = true;  for (int Index = 1; Result && (Index <= Buf.Length()); Index++)  {    wchar_t C = Buf[Index];    if (C == L'%')    {      break;    }    else if (C == L':')    {      Colons++;    }    else    {      Result = IsHex(C);    }  }  Result = Result && (Colons >= 2);  return Result;}//---------------------------------------------------------------------UnicodeString __fastcall EscapeIPv6Literal(const UnicodeString & IP){  UnicodeString Result = IP;  if (!HasIP6LiteralBrackets(Result))  {    Result = L"[" + IP + L"]";  }  return Result;}//---------------------------------------------------------------------TStrings * __fastcall TSessionData::GetRawSettingsForUrl(){  std::unique_ptr<TSessionData> FactoryDefaults(new TSessionData(L""));  std::unique_ptr<TSessionData> SessionData(Clone());  SessionData->FSProtocol = FactoryDefaults->FSProtocol;  SessionData->HostName = FactoryDefaults->HostName;  SessionData->PortNumber = FactoryDefaults->PortNumber;  SessionData->UserName = FactoryDefaults->UserName;  SessionData->Password = FactoryDefaults->Password;  SessionData->Ftps = FactoryDefaults->Ftps;  SessionData->HostKey = FactoryDefaults->HostKey;  SessionData->CopyNonCoreData(FactoryDefaults.get());  // Cannot be decided in SaveToOptions as it does not have HostName and UserName, so it cannot calculate DefaultSessionName.  bool SaveName = HasSessionName() && (Name != DefaultSessionName);  return SessionData->SaveToOptions(FactoryDefaults.get(), SaveName, false);}//---------------------------------------------------------------------bool __fastcall TSessionData::HasRawSettingsForUrl(){  std::unique_ptr<TStrings> RawSettings(GetRawSettingsForUrl());  return (RawSettings->Count > 0);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GenerateSessionUrl(unsigned int Flags){  UnicodeString Url;  if (FLAGSET(Flags, sufSpecific))  {    Url += WinSCPProtocolPrefix;  }  Url += GetProtocolUrl(FLAGSET(Flags, sufHttpForWebDAV));  // Add username only if it was somehow explicitly specified (so not with S3CredentialsEnv), but if it was, add it in the expanded form.  // For scripting, we might use unexpanded form (keeping the environment variables),  // but for consistency with code generation (where explicit expansion code would need to be added), we do not.  if (FLAGSET(Flags, sufUserName) && !UserName.IsEmpty())  {    Url += EncodeUrlString(UserNameExpanded);    if (FLAGSET(Flags, sufPassword) && !Password.IsEmpty())    {      Url += L":" + EncodeUrlString(NormalizeString(Password));    }    if (FLAGSET(Flags, sufHostKey) && !HostKey.IsEmpty())    {      UnicodeString KeyName;      UnicodeString Fingerprint = HostKey;      NormalizeFingerprint(Fingerprint, KeyName);      UnicodeString S = Fingerprint;      if (!KeyName.IsEmpty())      {        S = KeyName + NormalizedFingerprintSeparator + S;      }      S = Base64ToUrlSafe(S); // Noop for MD5 (both in SSH host keys and TLS/SSL)      S = MD5ToUrlSafe(S); // TLS/SSL fingerprints      UnicodeString S2 = EncodeUrlString(S);      DebugAssert(S2 == S2); // There should be nothing left for encoding      Url +=        UnicodeString(UrlParamSeparator) + UrlHostKeyParamName +        UnicodeString(UrlParamValueSeparator) + S2;    }    if (FLAGSET(Flags, sufRawSettings))    {      std::unique_ptr<TStrings> RawSettings(GetRawSettingsForUrl());      for (int Index = 0; Index < RawSettings->Count; Index++)      {        Url +=          UnicodeString(UrlParamSeparator) +          UrlRawSettingsParamNamePrefix + EncodeUrlString(LowerCase(RawSettings->Names[Index])) +          UnicodeString(UrlParamValueSeparator) + EncodeUrlString(RawSettings->ValueFromIndex[Index]);      }    }    Url += L"@";  }  DebugAssert(!HostNameExpanded.IsEmpty());  if (IsIPv6Literal(HostNameExpanded))  {    Url += EscapeIPv6Literal(HostNameExpanded);  }  else  {    Url += EncodeUrlString(HostNameExpanded);  }  if (PortNumber != GetDefaultPort())  {    Url += L":" + IntToStr(PortNumber);  }  Url += L"/";  return Url;}//---------------------------------------------------------------------UnicodeString ScriptCommandOpenLink(TraceInitStr(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);  }  if (Tunnel)  {    // not used anyway    int TunnelPortNumber = std::max(TunnelLocalPortNumber, Configuration->TunnelLocalPortNumberLow);    std::unique_ptr<TSessionData> TunnelData(CreateTunnelData(TunnelPortNumber));    TunnelHostKey = Configuration->LastFingerprint(TunnelData->SiteKey, SshFingerprintType);  }}//---------------------------------------------------------------------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(), false, false));  if (RawSettings->Count > 0)  {    AddSwitch(Result, RawSettingsOption, Rtf);    Result += StringsToParams(RawSettings.get());  }  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;    case fsS3:      ProtocolMember = "S3";      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;  }  int ADefaultPort = GetDefaultPort();  if (SessionData->PortNumber != ADefaultPort)  {    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", NormalizeString(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;      case fsS3:        // implicit        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;  }  Head += AssemblyNewClassInstanceEnd(Language, false);  std::unique_ptr<TStrings> RawSettings(SessionData->SaveToOptions(FactoryDefaults.get(), false, false));  UnicodeString SessionOptionsVariableName = AssemblyVariableName(Language, SessionOptionsClassName);  if (RawSettings->Count > 0)  {    Head +=      RtfPara +      AssemblyAddRawSettings(Language, RawSettings.get(), SessionOptionsClassName, L"AddRawSettings");  }  Head += RtfPara;  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 +=        RtfKeyword(L"using") + RtfText(" (") + NewSessionInstance + RtfText(L"())") + RtfPara +        RtfText(L"{") + RtfPara +        OpenCall;      Tail =        RtfText(L"}") + RtfPara;      break;    case alVBNET:      Head +=        RtfKeyword(L"Using") + RtfText(L" ") + NewSessionInstance + RtfPara +        OpenCall;      Tail =        RtfKeyword(L"End Using") + RtfPara;      break;    case alPowerShell:      Head +=        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);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetLocalDirectoryExpanded(){  return ExpandFileName(::ExpandEnvironmentVariables(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::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::SetSourceAddress(const UnicodeString & value){  SET_SESSION_PROPERTY(SourceAddress);}//---------------------------------------------------------------------void __fastcall TSessionData::SetProtocolFeatures(const UnicodeString & value){  SET_SESSION_PROPERTY(ProtocolFeatures);}//---------------------------------------------------------------------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::SetPuttySettings(UnicodeString value){  SET_SESSION_PROPERTY(PuttySettings);}//---------------------------------------------------------------------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::SetSFTPRealPath(TAutoSwitch value){  SET_SESSION_PROPERTY(SFTPRealPath);}//---------------------------------------------------------------------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::SetTunnelPassphrase(UnicodeString avalue){  RawByteString value = EncryptPassword(avalue, TunnelPublicKeyFile);  SET_SESSION_PROPERTY(TunnelPassphrase);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetTunnelPassphrase() const{  return DecryptPassword(FTunnelPassphrase, TunnelPublicKeyFile);}//---------------------------------------------------------------------void __fastcall TSessionData::SetTunnelPublicKeyFile(UnicodeString value){  if (FTunnelPublicKeyFile != value)  {    // TunnelPublicKeyFile is key for TunnelPassphrase encryption    UnicodeString XTunnelPassphrase = TunnelPassphrase;    // StripPathQuotes should not be needed as we do not feed quotes anymore    FTunnelPublicKeyFile = StripPathQuotes(value);    Modify();    TunnelPassphrase = XTunnelPassphrase;    Shred(XTunnelPassphrase);  }}//---------------------------------------------------------------------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::SetLogicalHostName(UnicodeString value){  SET_SESSION_PROPERTY(LogicalHostName);}//---------------------------------------------------------------------void __fastcall TSessionData::SetFtpListAll(TAutoSwitch value){  SET_SESSION_PROPERTY(FtpListAll);}//---------------------------------------------------------------------void __fastcall TSessionData::SetFtpHost(TAutoSwitch value){  SET_SESSION_PROPERTY(FtpHost);}//---------------------------------------------------------------------void __fastcall TSessionData::SetFtpWorkFromCwd(TAutoSwitch value){  SET_SESSION_PROPERTY(FtpWorkFromCwd);}//---------------------------------------------------------------------void TSessionData::SetFtpAnyCodeForPwd(bool value){  SET_SESSION_PROPERTY(FtpAnyCodeForPwd);}//---------------------------------------------------------------------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::SetInternalEditorEncoding(int value){  SET_SESSION_PROPERTY(InternalEditorEncoding);}//---------------------------------------------------------------------void __fastcall TSessionData::SetS3DefaultRegion(UnicodeString value){  SET_SESSION_PROPERTY(S3DefaultRegion);}//---------------------------------------------------------------------void __fastcall TSessionData::SetS3SessionToken(UnicodeString value){  SET_SESSION_PROPERTY(S3SessionToken);}//---------------------------------------------------------------------void __fastcall TSessionData::SetS3UrlStyle(TS3UrlStyle value){  SET_SESSION_PROPERTY(S3UrlStyle);}//---------------------------------------------------------------------void __fastcall TSessionData::SetS3MaxKeys(TAutoSwitch value){  SET_SESSION_PROPERTY(S3MaxKeys);}//---------------------------------------------------------------------void __fastcall TSessionData::SetS3CredentialsEnv(bool value){  SET_SESSION_PROPERTY(S3CredentialsEnv);}//---------------------------------------------------------------------void __fastcall TSessionData::SetIsWorkspace(bool value){  SET_SESSION_PROPERTY(IsWorkspace);}//---------------------------------------------------------------------void __fastcall TSessionData::SetLink(UnicodeString value){  SET_SESSION_PROPERTY(Link);}//---------------------------------------------------------------------void __fastcall TSessionData::SetNameOverride(UnicodeString value){  SET_SESSION_PROPERTY(NameOverride);}//---------------------------------------------------------------------void __fastcall TSessionData::SetHostKey(UnicodeString value){  SET_SESSION_PROPERTY(HostKey);}//---------------------------------------------------------------------void __fastcall TSessionData::SetNote(UnicodeString value){  SET_SESSION_PROPERTY(Note);}//---------------------------------------------------------------------void __fastcall TSessionData::SetWinTitle(UnicodeString value){  SET_SESSION_PROPERTY(WinTitle);}//---------------------------------------------------------------------UnicodeString __fastcall TSessionData::GetEncryptKey() const{  return DecryptPassword(FEncryptKey, UserName+HostName);}//---------------------------------------------------------------------void __fastcall TSessionData::SetEncryptKey(UnicodeString avalue){  RawByteString value = EncryptPassword(avalue, UserName+HostName);  SET_SESSION_PROPERTY(EncryptKey);}//---------------------------------------------------------------------void __fastcall TSessionData::SetWebDavLiberalEscaping(bool value){  SET_SESSION_PROPERTY(WebDavLiberalEscaping);}//---------------------------------------------------------------------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;}//---------------------------------------------------------------------void __fastcall TSessionData::DisableAuthentationsExceptPassword(){  SshNoUserAuth = false;  AuthKI = false;  AuthKIPassword = false;  AuthGSSAPI = false;  AuthGSSAPIKEX = false;  PublicKeyFile = L"";  TlsCertificateFile = L"";  Passphrase = L"";  TryAgent = false;}//---------------------------------------------------------------------TStrings * TSessionData::GetAllOptionNames(bool PuttyExport){  std::unique_ptr<TSessionData> FactoryDefaults(new TSessionData(L""));  return FactoryDefaults->SaveToOptions(NULL, false, PuttyExport);}//---------------------------------------------------------------------bool TSessionData::HasS3AutoCredentials(){  return (FSProtocol == fsS3) && S3CredentialsEnv;}//---------------------------------------------------------------------bool TSessionData::HasAutoCredentials(){  return HasS3AutoCredentials();}//=== TStoredSessionList ----------------------------------------------__fastcall TStoredSessionList::TStoredSessionList(bool aReadOnly):  TNamedObjectList(), FReadOnly(aReadOnly){  DebugAssert(Configuration);  FDefaultSettings = new TSessionData(DefaultName);  FPendingRemovals.reset(new TStringList());}//---------------------------------------------------------------------__fastcall TStoredSessionList::~TStoredSessionList(){  DebugAssert(Configuration);  delete FDefaultSettings;}//---------------------------------------------------------------------void __fastcall TStoredSessionList::Load(THierarchicalStorage * Storage,  bool AsModified, bool UseDefaults, bool PuttyImport){  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, PuttyImport);          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::Reload(){  if (Count <= Configuration->DontReloadMoreThanSessions)  {    bool SessionList = true;    std::unique_ptr<THierarchicalStorage> Storage(Configuration->CreateScpStorage(SessionList));    if (Storage->OpenSubKey(Configuration->StoredSessionsSubKey, False))    {      Load(Storage.get());    }  }}//---------------------------------------------------------------------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  {    while (FPendingRemovals->Count > 0)    {      TSessionData::Remove(Storage, FPendingRemovals->Strings[0]);      FPendingRemovals->Delete(0);    }    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, _di_IXMLNode SettingsNode){  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, SettingsNode);      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), SettingsNode);    }  }}//---------------------------------------------------------------------void __fastcall TStoredSessionList::ImportFromFilezilla(  const UnicodeString FileName, const UnicodeString ConfigurationFileName){  // not sure if the document must exists if we want to use its node  _di_IXMLDocument ConfigurationDocument;  _di_IXMLNode SettingsNode;  if (FileExists(ApiPath(ConfigurationFileName)))  {    ConfigurationDocument = interface_cast<Xmlintf::IXMLDocument>(new TXMLDocument(NULL));    ConfigurationDocument->LoadFromFile(ConfigurationFileName);    _di_IXMLNode FileZilla3Node = ConfigurationDocument->ChildNodes->FindNode(L"FileZilla3");    if (FileZilla3Node != NULL)    {      SettingsNode = FileZilla3Node->ChildNodes->FindNode(L"Settings");    }  }  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"", SettingsNode);    }  }}//---------------------------------------------------------------------UnicodeString FormatKnownHostName(const UnicodeString & HostName, int PortNumber){  return FORMAT(L"%s:%d", (HostName, PortNumber));}//---------------------------------------------------------------------void __fastcall TStoredSessionList::ImportFromKnownHosts(TStrings * Lines){  bool SessionList = false;  std::unique_ptr<THierarchicalStorage> HostKeyStorage(Configuration->CreateScpStorage(SessionList));  std::unique_ptr<TStrings> KeyList(new TStringList());  if (OpenHostKeysSubKey(HostKeyStorage.get(), false))  {    HostKeyStorage->GetValueNames(KeyList.get());  }  HostKeyStorage.reset(NULL);  UnicodeString FirstError;  for (int Index = 0; Index < Lines->Count; Index++)  {    try    {      UnicodeString Line = Lines->Strings[Index];      Line = Trim(Line);      if (IsValidOpensshLine(Line))      {        int P = Pos(L' ', Line);        if (P > 0)        {          UnicodeString HostNameStr = Line.SubString(1, P - 1);          Line = Line.SubString(P + 1, Line.Length() - P);          P = Pos(L',', HostNameStr);          if (P > 0)          {            HostNameStr.SetLength(P - 1);          }          P = Pos(L':', HostNameStr);          int PortNumber = -1;          if (P > 0)          {            UnicodeString PortNumberStr = HostNameStr.SubString(P + 1, HostNameStr.Length() - P);            PortNumber = StrToInt(PortNumberStr);            HostNameStr.SetLength(P - 1);          }          if (HasIP6LiteralBrackets(HostNameStr))          {            HostNameStr = StripIP6LiteralBrackets(HostNameStr);          }          UnicodeString NameStr = HostNameStr;          if (PortNumber >= 0)          {            NameStr = FormatKnownHostName(NameStr, PortNumber);          }          std::unique_ptr<TSessionData> SessionDataOwner;          TSessionData * SessionData = dynamic_cast<TSessionData *>(FindByName(NameStr));          if (SessionData == NULL)          {            SessionData = new TSessionData(L"");            SessionDataOwner.reset(SessionData);            SessionData->CopyData(DefaultSettings);            SessionData->Name = NameStr;            SessionData->HostName = HostNameStr;            if (PortNumber >= 0)            {              SessionData->PortNumber = PortNumber;            }          }          const struct ssh_keyalg * Algorithm;          UnicodeString Key = ParseOpenSshPubLine(Line, Algorithm);          UnicodeString KeyKey =            FORMAT(L"%s@%d:%s", (Algorithm->cache_id, SessionData->PortNumber, HostNameStr));          UnicodeString HostKey =            FORMAT(L"%s:%s=%s", (Algorithm->ssh_id, KeyKey, Key));          UnicodeString HostKeyList = SessionData->HostKey;          AddToList(HostKeyList, HostKey, L";");          SessionData->HostKey = HostKeyList;          // If there's at least one unknown key type for this host, select it          if (KeyList->IndexOf(KeyKey) < 0)          {            SessionData->Selected = true;          }          if (SessionDataOwner.get() != NULL)          {            Add(SessionDataOwner.release());          }        }      }    }    catch (Exception & E)    {      if (FirstError.IsEmpty())      {        FirstError = E.Message;      }    }  }  if (Count == 0)  {    UnicodeString Message = LoadStr(KNOWN_HOSTS_NO_SITES);    if (!FirstError.IsEmpty())    {      Message = FORMAT(L"%s\n(%s)", (Message, FirstError));    }    throw Exception(Message);  }}//---------------------------------------------------------------------void TStoredSessionList::ImportFromOpenssh(TStrings * Lines){  std::unique_ptr<TStrings> Hosts(CreateSortedStringList());  for (int Index = 0; Index < Lines->Count; Index++)  {    UnicodeString Line = Lines->Strings[Index];    UnicodeString Directive, Value;    if (ParseOpensshDirective(Line, Directive, Value) &&        SameText(Directive, OpensshHostDirective))    {      while (!Value.IsEmpty())      {        UnicodeString Name = CutOpensshToken(Value);        if ((Hosts->IndexOf(Name) < 0) && (Name.LastDelimiter(L"*?") == 0))        {          std::unique_ptr<TSessionData> Data(new TSessionData(EmptyStr));          Data->CopyData(DefaultSettings);          Data->Name = Name;          Data->HostName = Name;          Data->ImportFromOpenssh(Lines);          Add(Data.release());          Hosts->Add(Name);        }      }    }  }}//---------------------------------------------------------------------void __fastcall TStoredSessionList::Export(const UnicodeString FileName){  THierarchicalStorage * Storage = TIniFileStorage::CreateFromPath(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 S3 = 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;        case fsS3:          S3++;          break;      }      if (Data->HasAnySessionPassword())      {        Password++;      }      if (Data->Color != 0)      {        Color++;      }      if (!Data->Note.IsEmpty())      {        Note++;      }      // This would not work for passwords, as they are compared in their encrypted form.      // But there are no passwords set in factory defaults anyway.      if (!Data->IsSame(FactoryDefaults.get(), true, DifferentAdvancedProperties.get(), false))      {        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"StoredSessionsCountS3", S3);  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);    }  }}//---------------------------------------------------------------------------bool __fastcall TStoredSessionList::OpenHostKeysSubKey(THierarchicalStorage * Storage, bool CanCreate){  return    Storage->OpenRootKey(CanCreate) &&    Storage->OpenSubKey(Configuration->SshHostKeysSubKey, CanCreate);}//---------------------------------------------------------------------------THierarchicalStorage * __fastcall TStoredSessionList::CreateHostKeysStorageForWritting(){  bool SessionList = false;  std::unique_ptr<THierarchicalStorage> Storage(Configuration->CreateScpStorage(SessionList));  Storage->Explicit = true;  Storage->AccessMode = smReadWrite;  return Storage.release();}//---------------------------------------------------------------------------void __fastcall TStoredSessionList::ImportHostKeys(  THierarchicalStorage * SourceStorage, THierarchicalStorage * TargetStorage, TStoredSessionList * Sessions, bool OnlySelected){  if (OpenHostKeysSubKey(SourceStorage, false) &&      OpenHostKeysSubKey(TargetStorage, true))  {    std::auto_ptr<TStringList> KeyList(new TStringList());    SourceStorage->GetValueNames(KeyList.get());    DebugAssert(Sessions != NULL);    for (int Index = 0; Index < Sessions->Count; Index++)    {      TSessionData * Session = Sessions->Sessions[Index];      if (!OnlySelected || Session->Selected)      {        UnicodeString HostKeyName = PuttyMungeStr(FORMAT(L"@%d:%s", (Session->PortNumber, Session->HostNameExpanded)));        for (int KeyIndex = 0; KeyIndex < KeyList->Count; KeyIndex++)        {          UnicodeString KeyName = KeyList->Strings[KeyIndex];          if (EndsText(HostKeyName, KeyName))          {            TargetStorage->WriteStringRaw(KeyName, SourceStorage->ReadStringRaw(KeyName, L""));          }        }      }    }  }}//---------------------------------------------------------------------------void __fastcall TStoredSessionList::ImportHostKeys(  const UnicodeString & SourceKey, TStoredSessionList * Sessions, bool OnlySelected){  std::auto_ptr<THierarchicalStorage> TargetStorage(CreateHostKeysStorageForWritting());  std::auto_ptr<THierarchicalStorage> SourceStorage(new TRegistryStorage(SourceKey));  ImportHostKeys(SourceStorage.get(), TargetStorage.get(), Sessions, OnlySelected);}//---------------------------------------------------------------------------void __fastcall TStoredSessionList::ImportSelectedKnownHosts(TStoredSessionList * Sessions){  std::unique_ptr<THierarchicalStorage> Storage(CreateHostKeysStorageForWritting());  if (OpenHostKeysSubKey(Storage.get(), true))  {    for (int Index = 0; Index < Sessions->Count; Index++)    {      TSessionData * Session = Sessions->Sessions[Index];      if (Session->Selected)      {        UnicodeString Algs;        UnicodeString HostKeys = Session->HostKey;        while (!HostKeys.IsEmpty())        {          UnicodeString HostKey = CutToChar(HostKeys, L';', true);          // skip alg          CutToChar(HostKey, L':', true);          UnicodeString Key = CutToChar(HostKey, L'=', true);          Storage->WriteStringRaw(Key, HostKey);        }      }    }  }}//---------------------------------------------------------------------------void TStoredSessionList::SelectKnownHostsForSelectedSessions(  TStoredSessionList * KnownHosts, TStoredSessionList * Sessions){  for (int SessionIndex = 0; SessionIndex < Sessions->Count; SessionIndex++)  {    TSessionData * Session = Sessions->Sessions[SessionIndex];    if (Session->Selected)    {      UnicodeString Key = Session->HostName;      if (Session->PortNumber != Session->GetDefaultPort())      {        Key = FormatKnownHostName(Key, Session->PortNumber);      }      TSessionData * KnownHost = dynamic_cast<TSessionData *>(KnownHosts->FindByName(Key));      if (KnownHost != NULL)      {        KnownHost->Selected = true;      }    }  }}//---------------------------------------------------------------------------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){  DoGetFolderOrWorkspace(Name, List, false);}//---------------------------------------------------------------------------void __fastcall TStoredSessionList::DoGetFolderOrWorkspace(const UnicodeString & Name, TList * List, bool NoRecrypt){  for (int Index = 0; (Index < Count); Index++)  {    TSessionData * RawData = Sessions[Index];    TSessionData * Data =      CheckIsInFolderOrWorkspaceAndResolve(RawData, Name);    if (Data != NULL)    {      TSessionData * Data2 = new TSessionData(L"");      if (NoRecrypt)      {        Data2->CopyDataNoRecrypt(Data);      }      else      {        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);      }      if (!RawData->NameOverride.IsEmpty())      {        Data2->Name = RawData->NameOverride;      }      else if (RawData->Link.IsEmpty() && RawData->IsWorkspace)      {        // Newly opened ad-hoc session has no name, so restore the workspace that way too.        // Otherwise we would persist the generated internal workspace name as a real name.        Data2->Name = UnicodeString();      }      List->Add(Data2);    }  }}//---------------------------------------------------------------------------TStrings * __fastcall TStoredSessionList::GetFolderOrWorkspaceList(  const UnicodeString & Name){  std::unique_ptr<TObjectList> DataList(new TObjectList());  DoGetFolderOrWorkspace(Name, DataList.get(), true);  std::unique_ptr<TStringList> Result(new TStringList());  for (int Index = 0; (Index < DataList->Count); Index++)  {    Result->Add(dynamic_cast<TSessionData *>(DataList->Items[Index])->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))    {      FPendingRemovals->Add(Data->Name);      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, int Flags){  TSessionData * Data = new TSessionData(L"");  try  {    Data->ParseUrl(Url, Options, this, DefaultsOnly, FileName, AProtocolDefined, MaskedUrl, Flags);  }  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, int Index){  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->NameOverride = Data->Name;  }  Result->IsWorkspace = true;  Result->Name = IntToHex(Index, 4); // See HasSessionName()  return Result.release();}//---------------------------------------------------------------------bool __fastcall TStoredSessionList::CanLogin(TSessionData * Data){  Data = ResolveWorkspaceData(Data);  return (Data != NULL) && Data->CanLogin;}//---------------------------------------------------------------------UnicodeString GetExpandedLogFileName(UnicodeString LogFileName, TDateTime Started, TSessionData * SessionData){  // StripPathQuotes should not be needed as we do not feed quotes anymore  UnicodeString ANewFileName = StripPathQuotes(ExpandEnvironmentVariables(LogFileName));  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", Started);          break;        case L'm':          Replacement = FormatDateTime(L"mm", Started);          break;        case L'd':          Replacement = FormatDateTime(L"dd", Started);          break;        case L't':          Replacement = FormatDateTime(L"hhnnss", Started);          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:    case fsS3:      if (Ftps == ftpsNone)      {        Result = HTTPPortNumber;      }      else      {        Result = HTTPSPortNumber;      }      break;    default:      if (IsSshProtocol(FSProtocol))      {        Result = SshPortNumber;      }      else      {        DebugFail();        Result = -1;      }      break;  }  return Result;}//---------------------------------------------------------------------
 |