workunit.cpp 345 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762676367646765676667676768676967706771677267736774677567766777677867796780678167826783678467856786678767886789679067916792679367946795679667976798679968006801680268036804680568066807680868096810681168126813681468156816681768186819682068216822682368246825682668276828682968306831683268336834683568366837683868396840684168426843684468456846684768486849685068516852685368546855685668576858685968606861686268636864686568666867686868696870687168726873687468756876687768786879688068816882688368846885688668876888688968906891689268936894689568966897689868996900690169026903690469056906690769086909691069116912691369146915691669176918691969206921692269236924692569266927692869296930693169326933693469356936693769386939694069416942694369446945694669476948694969506951695269536954695569566957695869596960696169626963696469656966696769686969697069716972697369746975697669776978697969806981698269836984698569866987698869896990699169926993699469956996699769986999700070017002700370047005700670077008700970107011701270137014701570167017701870197020702170227023702470257026702770287029703070317032703370347035703670377038703970407041704270437044704570467047704870497050705170527053705470557056705770587059706070617062706370647065706670677068706970707071707270737074707570767077707870797080708170827083708470857086708770887089709070917092709370947095709670977098709971007101710271037104710571067107710871097110711171127113711471157116711771187119712071217122712371247125712671277128712971307131713271337134713571367137713871397140714171427143714471457146714771487149715071517152715371547155715671577158715971607161716271637164716571667167716871697170717171727173717471757176717771787179718071817182718371847185718671877188718971907191719271937194719571967197719871997200720172027203720472057206720772087209721072117212721372147215721672177218721972207221722272237224722572267227722872297230723172327233723472357236723772387239724072417242724372447245724672477248724972507251725272537254725572567257725872597260726172627263726472657266726772687269727072717272727372747275727672777278727972807281728272837284728572867287728872897290729172927293729472957296729772987299730073017302730373047305730673077308730973107311731273137314731573167317731873197320732173227323732473257326732773287329733073317332733373347335733673377338733973407341734273437344734573467347734873497350735173527353735473557356735773587359736073617362736373647365736673677368736973707371737273737374737573767377737873797380738173827383738473857386738773887389739073917392739373947395739673977398739974007401740274037404740574067407740874097410741174127413741474157416741774187419742074217422742374247425742674277428742974307431743274337434743574367437743874397440744174427443744474457446744774487449745074517452745374547455745674577458745974607461746274637464746574667467746874697470747174727473747474757476747774787479748074817482748374847485748674877488748974907491749274937494749574967497749874997500750175027503750475057506750775087509751075117512751375147515751675177518751975207521752275237524752575267527752875297530753175327533753475357536753775387539754075417542754375447545754675477548754975507551755275537554755575567557755875597560756175627563756475657566756775687569757075717572757375747575757675777578757975807581758275837584758575867587758875897590759175927593759475957596759775987599760076017602760376047605760676077608760976107611761276137614761576167617761876197620762176227623762476257626762776287629763076317632763376347635763676377638763976407641764276437644764576467647764876497650765176527653765476557656765776587659766076617662766376647665766676677668766976707671767276737674767576767677767876797680768176827683768476857686768776887689769076917692769376947695769676977698769977007701770277037704770577067707770877097710771177127713771477157716771777187719772077217722772377247725772677277728772977307731773277337734773577367737773877397740774177427743774477457746774777487749775077517752775377547755775677577758775977607761776277637764776577667767776877697770777177727773777477757776777777787779778077817782778377847785778677877788778977907791779277937794779577967797779877997800780178027803780478057806780778087809781078117812781378147815781678177818781978207821782278237824782578267827782878297830783178327833783478357836783778387839784078417842784378447845784678477848784978507851785278537854785578567857785878597860786178627863786478657866786778687869787078717872787378747875787678777878787978807881788278837884788578867887788878897890789178927893789478957896789778987899790079017902790379047905790679077908790979107911791279137914791579167917791879197920792179227923792479257926792779287929793079317932793379347935793679377938793979407941794279437944794579467947794879497950795179527953795479557956795779587959796079617962796379647965796679677968796979707971797279737974797579767977797879797980798179827983798479857986798779887989799079917992799379947995799679977998799980008001800280038004800580068007800880098010801180128013801480158016801780188019802080218022802380248025802680278028802980308031803280338034803580368037803880398040804180428043804480458046804780488049805080518052805380548055805680578058805980608061806280638064806580668067806880698070807180728073807480758076807780788079808080818082808380848085808680878088808980908091809280938094809580968097809880998100810181028103810481058106810781088109811081118112811381148115811681178118811981208121812281238124812581268127812881298130813181328133813481358136813781388139814081418142814381448145814681478148814981508151815281538154815581568157815881598160816181628163816481658166816781688169817081718172817381748175817681778178817981808181818281838184818581868187818881898190819181928193819481958196819781988199820082018202820382048205820682078208820982108211821282138214821582168217821882198220822182228223822482258226822782288229823082318232823382348235823682378238823982408241824282438244824582468247824882498250825182528253825482558256825782588259826082618262826382648265826682678268826982708271827282738274827582768277827882798280828182828283828482858286828782888289829082918292829382948295829682978298829983008301830283038304830583068307830883098310831183128313831483158316831783188319832083218322832383248325832683278328832983308331833283338334833583368337833883398340834183428343834483458346834783488349835083518352835383548355835683578358835983608361836283638364836583668367836883698370837183728373837483758376837783788379838083818382838383848385838683878388838983908391839283938394839583968397839883998400840184028403840484058406840784088409841084118412841384148415841684178418841984208421842284238424842584268427842884298430843184328433843484358436843784388439844084418442844384448445844684478448844984508451845284538454845584568457845884598460846184628463846484658466846784688469847084718472847384748475847684778478847984808481848284838484848584868487848884898490849184928493849484958496849784988499850085018502850385048505850685078508850985108511851285138514851585168517851885198520852185228523852485258526852785288529853085318532853385348535853685378538853985408541854285438544854585468547854885498550855185528553855485558556855785588559856085618562856385648565856685678568856985708571857285738574857585768577857885798580858185828583858485858586858785888589859085918592859385948595859685978598859986008601860286038604860586068607860886098610861186128613861486158616861786188619862086218622862386248625862686278628862986308631863286338634863586368637863886398640864186428643864486458646864786488649865086518652865386548655865686578658865986608661866286638664866586668667866886698670867186728673867486758676867786788679868086818682868386848685868686878688868986908691869286938694869586968697869886998700870187028703870487058706870787088709871087118712871387148715871687178718871987208721872287238724872587268727872887298730873187328733873487358736873787388739874087418742874387448745874687478748874987508751875287538754875587568757875887598760876187628763876487658766876787688769877087718772877387748775877687778778877987808781878287838784878587868787878887898790879187928793879487958796879787988799880088018802880388048805880688078808880988108811881288138814881588168817881888198820882188228823882488258826882788288829883088318832883388348835883688378838883988408841884288438844884588468847884888498850885188528853885488558856885788588859886088618862886388648865886688678868886988708871887288738874887588768877887888798880888188828883888488858886888788888889889088918892889388948895889688978898889989008901890289038904890589068907890889098910891189128913891489158916891789188919892089218922892389248925892689278928892989308931893289338934893589368937893889398940894189428943894489458946894789488949895089518952895389548955895689578958895989608961896289638964896589668967896889698970897189728973897489758976897789788979898089818982898389848985898689878988898989908991899289938994899589968997899889999000900190029003900490059006900790089009901090119012901390149015901690179018901990209021902290239024902590269027902890299030903190329033903490359036903790389039904090419042904390449045904690479048904990509051905290539054905590569057905890599060906190629063906490659066906790689069907090719072907390749075907690779078907990809081908290839084908590869087908890899090909190929093909490959096909790989099910091019102910391049105910691079108910991109111911291139114911591169117911891199120912191229123912491259126912791289129913091319132913391349135913691379138913991409141914291439144914591469147914891499150915191529153915491559156915791589159916091619162916391649165916691679168916991709171917291739174917591769177917891799180918191829183918491859186918791889189919091919192919391949195919691979198919992009201920292039204920592069207920892099210921192129213921492159216921792189219922092219222922392249225922692279228922992309231923292339234923592369237923892399240924192429243924492459246924792489249925092519252925392549255925692579258925992609261926292639264926592669267926892699270927192729273927492759276927792789279928092819282928392849285928692879288928992909291929292939294929592969297929892999300930193029303930493059306930793089309931093119312931393149315931693179318931993209321932293239324932593269327932893299330933193329333933493359336933793389339934093419342934393449345934693479348934993509351935293539354935593569357935893599360936193629363936493659366936793689369937093719372937393749375937693779378937993809381938293839384938593869387938893899390939193929393939493959396939793989399940094019402940394049405940694079408940994109411941294139414941594169417941894199420942194229423942494259426942794289429943094319432943394349435943694379438943994409441944294439444944594469447944894499450945194529453945494559456945794589459946094619462946394649465946694679468946994709471947294739474947594769477947894799480948194829483948494859486948794889489949094919492949394949495949694979498949995009501950295039504950595069507950895099510951195129513951495159516951795189519952095219522952395249525952695279528952995309531953295339534953595369537953895399540954195429543954495459546954795489549955095519552955395549555955695579558955995609561956295639564956595669567956895699570957195729573957495759576957795789579958095819582958395849585958695879588958995909591959295939594959595969597959895999600960196029603960496059606960796089609961096119612961396149615961696179618961996209621962296239624962596269627962896299630963196329633963496359636963796389639964096419642964396449645964696479648964996509651965296539654965596569657965896599660966196629663966496659666966796689669967096719672967396749675967696779678967996809681968296839684968596869687968896899690969196929693969496959696969796989699970097019702970397049705970697079708970997109711971297139714971597169717971897199720972197229723972497259726972797289729973097319732973397349735973697379738973997409741974297439744974597469747974897499750975197529753975497559756975797589759976097619762976397649765976697679768976997709771977297739774977597769777977897799780978197829783978497859786978797889789979097919792979397949795979697979798979998009801980298039804980598069807980898099810981198129813981498159816981798189819982098219822982398249825982698279828982998309831983298339834983598369837983898399840984198429843984498459846984798489849985098519852985398549855985698579858985998609861986298639864986598669867986898699870987198729873987498759876987798789879988098819882988398849885988698879888988998909891989298939894989598969897989898999900990199029903990499059906990799089909991099119912991399149915991699179918991999209921992299239924992599269927992899299930993199329933993499359936993799389939994099419942994399449945994699479948994999509951995299539954995599569957995899599960996199629963996499659966996799689969997099719972997399749975997699779978997999809981998299839984998599869987998899899990999199929993999499959996999799989999100001000110002100031000410005100061000710008100091001010011100121001310014100151001610017100181001910020100211002210023100241002510026100271002810029100301003110032100331003410035100361003710038100391004010041100421004310044100451004610047100481004910050100511005210053100541005510056100571005810059100601006110062100631006410065100661006710068100691007010071100721007310074100751007610077100781007910080100811008210083100841008510086100871008810089100901009110092100931009410095100961009710098100991010010101101021010310104101051010610107101081010910110101111011210113101141011510116101171011810119101201012110122101231012410125101261012710128101291013010131101321013310134101351013610137101381013910140101411014210143101441014510146101471014810149101501015110152101531015410155101561015710158101591016010161101621016310164101651016610167101681016910170101711017210173101741017510176101771017810179101801018110182101831018410185101861018710188101891019010191101921019310194101951019610197101981019910200102011020210203102041020510206102071020810209102101021110212102131021410215102161021710218102191022010221102221022310224102251022610227102281022910230102311023210233102341023510236102371023810239102401024110242102431024410245102461024710248102491025010251102521025310254102551025610257102581025910260102611026210263102641026510266102671026810269102701027110272102731027410275102761027710278102791028010281102821028310284102851028610287102881028910290102911029210293102941029510296102971029810299103001030110302103031030410305103061030710308103091031010311103121031310314103151031610317103181031910320103211032210323103241032510326103271032810329103301033110332103331033410335103361033710338103391034010341103421034310344103451034610347103481034910350103511035210353103541035510356103571035810359103601036110362103631036410365103661036710368103691037010371103721037310374103751037610377103781037910380103811038210383103841038510386103871038810389103901039110392103931039410395103961039710398103991040010401104021040310404104051040610407104081040910410104111041210413104141041510416104171041810419104201042110422104231042410425104261042710428104291043010431104321043310434104351043610437104381043910440104411044210443104441044510446104471044810449104501045110452104531045410455104561045710458104591046010461104621046310464104651046610467104681046910470104711047210473104741047510476104771047810479104801048110482104831048410485104861048710488104891049010491104921049310494104951049610497104981049910500105011050210503105041050510506105071050810509105101051110512105131051410515105161051710518105191052010521105221052310524105251052610527105281052910530105311053210533105341053510536105371053810539105401054110542105431054410545105461054710548105491055010551105521055310554105551055610557105581055910560105611056210563105641056510566105671056810569105701057110572105731057410575105761057710578105791058010581105821058310584105851058610587105881058910590105911059210593105941059510596105971059810599106001060110602106031060410605106061060710608106091061010611106121061310614106151061610617106181061910620106211062210623
  1. /*##############################################################################
  2. HPCC SYSTEMS software Copyright (C) 2012 HPCC Systems.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. ############################################################################## */
  13. #include "jlib.hpp"
  14. #include "workunit.hpp"
  15. #include "jprop.hpp"
  16. #include "jmisc.hpp"
  17. #include "jexcept.hpp"
  18. #include "jiter.ipp"
  19. #include "jptree.hpp"
  20. #include "jtime.ipp"
  21. #include "jencrypt.hpp"
  22. #include "junicode.hpp"
  23. #include "jlzw.hpp"
  24. #include "jregexp.hpp"
  25. #include "eclrtl.hpp"
  26. #include "deftype.hpp"
  27. #include <time.h>
  28. #include "mpbase.hpp"
  29. #include "daclient.hpp"
  30. #include "dadfs.hpp"
  31. #include "dafdesc.hpp"
  32. #include "dasds.hpp"
  33. #include "danqs.hpp"
  34. #include "dautils.hpp"
  35. #include "dllserver.hpp"
  36. #include "thorplugin.hpp"
  37. #include "thorhelper.hpp"
  38. #include "workflow.hpp"
  39. #include "nbcd.hpp"
  40. #include "seclib.hpp"
  41. #include "wuerror.hpp"
  42. #include "wujobq.hpp"
  43. #include "environment.hpp"
  44. #include "workunit.ipp"
  45. #define GLOBAL_WORKUNIT "global"
  46. #define SDS_LOCK_TIMEOUT (5*60*1000) // 5mins, 30s a bit short
  47. static int workUnitTraceLevel = 1;
  48. static StringBuffer &getXPath(StringBuffer &wuRoot, const char *wuid)
  49. {
  50. // MORE - can fold in the date
  51. return wuRoot.append("/WorkUnits/").append(wuid);
  52. }
  53. //To be called by eclserver, but esp etc. won't know, so we need to store it.
  54. static StringBuffer & appendLibrarySuffix(StringBuffer & suffix)
  55. {
  56. #ifdef _WIN32
  57. suffix.append("W");
  58. #else
  59. suffix.append("L");
  60. #endif
  61. #ifdef __64BIT__
  62. suffix.append("64");
  63. #else
  64. suffix.append("32");
  65. #endif
  66. return suffix;
  67. }
  68. typedef MapStringTo<bool> UniqueScopes;
  69. static void wuAccessError(const char *username, const char *action, const char *wuscope, const char *wuid, bool excpt, bool log)
  70. {
  71. StringBuffer err;
  72. err.append("Workunit Access Denied - action: ").append(action).append(" user:").append(username ? username : "<Unknown>");
  73. if (wuid)
  74. err.append(" workunit:").append(wuid);
  75. if (wuscope)
  76. err.append(" scope:").append(wuscope);
  77. //MORE - we would need more information passed in from outside if we want to make the audit message format the same as from higher level ESP calls
  78. SYSLOG(AUDIT_TYPE_ACCESS_FAILURE, err.str());
  79. if (log)
  80. LOG(MCuserError, "%s", err.str());
  81. if (excpt)
  82. throw MakeStringException(WUERR_AccessError, "%s", err.str());
  83. }
  84. static bool checkWuScopeSecAccess(const char *wuscope, ISecManager *secmgr, ISecUser *secuser, int required, const char *action, bool excpt, bool log)
  85. {
  86. if (!secmgr || !secuser)
  87. return true;
  88. bool ret = secmgr->authorizeEx(RT_WORKUNIT_SCOPE, *secuser, wuscope)>=required;
  89. if (!ret && (log || excpt))
  90. wuAccessError(secuser->getName(), action, wuscope, NULL, excpt, log);
  91. return ret;
  92. }
  93. static bool checkWuScopeListSecAccess(const char *wuscope, ISecResourceList *scopes, int required, const char *action, bool excpt, bool log)
  94. {
  95. if (!scopes)
  96. return true;
  97. bool ret=true;
  98. if (wuscope)
  99. {
  100. Owned<ISecResource> res=scopes->getResource(wuscope);
  101. if (!res || res->getAccessFlags()<required)
  102. ret=false;
  103. }
  104. else
  105. {
  106. for (int seq=0; ret && seq<scopes->count(); seq++)
  107. {
  108. ISecResource *res=scopes->queryResource(seq);
  109. if (res && res->getAccessFlags()<required)
  110. return false;
  111. }
  112. }
  113. if (!ret && (log || excpt))
  114. wuAccessError(NULL, action, wuscope, NULL, excpt, log);
  115. return ret;
  116. }
  117. static bool checkWuSecAccess(IConstWorkUnit &cw, ISecManager *secmgr, ISecUser *secuser, int required, const char *action, bool excpt, bool log)
  118. {
  119. if (!secmgr || !secuser)
  120. return true;
  121. bool ret=secmgr->authorizeEx(RT_WORKUNIT_SCOPE, *secuser, cw.queryWuScope())>=required;
  122. if (!ret && (log || excpt))
  123. {
  124. wuAccessError(secuser->getName(), action, cw.queryWuScope(), cw.queryWuid(), excpt, log);
  125. }
  126. return ret;
  127. }
  128. static bool checkWuSecAccess(const char *wuid, ISecManager *secmgr, ISecUser *secuser, int required, const char *action, bool excpt, bool log)
  129. {
  130. StringBuffer wuRoot;
  131. Owned<IRemoteConnection> conn = querySDS().connect(getXPath(wuRoot, wuid).str(), myProcessSession(), 0, SDS_LOCK_TIMEOUT);
  132. if (conn)
  133. {
  134. Owned<IPropertyTree> ptree=conn->getRoot();
  135. return checkWuScopeSecAccess(ptree->queryProp("@scope"), secmgr, secuser, required, action, excpt, log);
  136. }
  137. if (log || excpt)
  138. wuAccessError(secuser ? secuser->getName() : NULL, action, "Unknown", NULL, excpt, log);
  139. return false;
  140. }
  141. void doDescheduleWorkkunit(char const * wuid)
  142. {
  143. StringBuffer xpath;
  144. xpath.append("*/*/*/");
  145. ncnameEscape(wuid, xpath);
  146. Owned<IRemoteConnection> conn = querySDS().connect("/Schedule", myProcessSession(), RTM_LOCK_WRITE, SDS_LOCK_TIMEOUT);
  147. if(!conn) return;
  148. Owned<IPropertyTree> root = conn->getRoot();
  149. bool more;
  150. do more = root->removeProp(xpath.str()); while(more);
  151. }
  152. #define PROGRESS_FORMAT_V 2
  153. class CConstGraphProgress : public CInterface, implements IConstWUGraphProgress
  154. {
  155. class CWuGraphStats : public CInterfaceOf<IWUGraphStats>
  156. {
  157. public:
  158. CWuGraphStats(CConstGraphProgress &_parent, StatisticCreatorType _creatorType, const char * _creator, const char * _rootScope, unsigned _id)
  159. : parent(_parent), creatorType(_creatorType), creator(_creator), id(_id)
  160. {
  161. StringBuffer subgraphScopeName;
  162. subgraphScopeName.append(_rootScope);
  163. StatisticScopeType scopeType = SSTgraph;
  164. StatsScopeId rootScopeId;
  165. verifyex(rootScopeId.setScopeText(_rootScope));
  166. collector.setown(createStatisticsGatherer(_creatorType, _creator, rootScopeId));
  167. }
  168. virtual void beforeDispose()
  169. {
  170. Owned<IStatisticCollection> stats = collector->getResult();
  171. MemoryBuffer compressed;
  172. {
  173. MemoryBuffer serialized;
  174. serializeStatisticCollection(serialized, stats);
  175. compressToBuffer(compressed, serialized.length(), serialized.toByteArray());
  176. }
  177. unsigned minActivity = 0;
  178. unsigned maxActivity = 0;
  179. stats->getMinMaxActivity(minActivity, maxActivity);
  180. parent.setSubgraphStats(creatorType, creator, id, compressed, minActivity, maxActivity);
  181. }
  182. virtual IStatisticGatherer & queryStatsBuilder()
  183. {
  184. return *collector;
  185. }
  186. protected:
  187. CConstGraphProgress &parent;
  188. Owned<IStatisticGatherer> collector;
  189. StringAttr creator;
  190. StatisticCreatorType creatorType;
  191. unsigned id;
  192. };
  193. class CGraphProgress : public CInterface, implements IWUGraphProgress
  194. {
  195. CConstGraphProgress &parent;
  196. public:
  197. IMPLEMENT_IINTERFACE;
  198. CGraphProgress(CConstGraphProgress &_parent) : parent(_parent)
  199. {
  200. parent.lockWrite();
  201. }
  202. ~CGraphProgress()
  203. {
  204. parent.unlock();
  205. }
  206. virtual IPropertyTree * getProgressTree() { return parent.getProgressTree(); }
  207. virtual WUGraphState queryGraphState() { return parent.queryGraphState(); }
  208. virtual WUGraphState queryNodeState(WUGraphIDType nodeId) { return parent.queryNodeState(nodeId); }
  209. virtual IWUGraphProgress * update() { throwUnexpected(); }
  210. virtual IWUGraphStats * update(StatisticCreatorType creatorType, const char * creator, unsigned subgraph) { throwUnexpected(); }
  211. virtual unsigned queryFormatVersion() { return parent.queryFormatVersion(); }
  212. virtual void setGraphState(WUGraphState state)
  213. {
  214. parent.setGraphState(state);
  215. }
  216. virtual void setNodeState(WUGraphIDType nodeId, WUGraphState state)
  217. {
  218. parent.setNodeState(nodeId, state);
  219. }
  220. };
  221. void clearConnection()
  222. {
  223. conn.clear();
  224. progress.clear();
  225. }
  226. public:
  227. IMPLEMENT_IINTERFACE;
  228. static void deleteWuidProgress(const char *wuid)
  229. {
  230. StringBuffer path("/GraphProgress/");
  231. path.append(wuid);
  232. Owned<IRemoteConnection> conn = querySDS().connect(path.str(), myProcessSession(), RTM_LOCK_WRITE, SDS_LOCK_TIMEOUT);
  233. if (conn)
  234. conn->close(true);
  235. }
  236. CConstGraphProgress(const char *_wuid, const char *_graphName) : wuid(_wuid), graphName(_graphName)
  237. {
  238. rootPath.append("/GraphProgress/").append(wuid).append('/').append(graphName).append('/');
  239. connected = connectedWrite = false;
  240. formatVersion = 0;
  241. }
  242. CConstGraphProgress(const char *_wuid, const char *_graphName, IPropertyTree *_progress) : wuid(_wuid), graphName(_graphName), progress(_progress)
  243. {
  244. formatVersion = progress->getPropInt("@format");
  245. connectedWrite = false; // should never be
  246. connected = true;
  247. }
  248. void connect()
  249. {
  250. clearConnection();
  251. packProgress(wuid,false);
  252. conn.setown(querySDS().connect(rootPath.str(), myProcessSession(), RTM_LOCK_READ|RTM_CREATE_QUERY, SDS_LOCK_TIMEOUT));
  253. progress.set(conn->queryRoot());
  254. formatVersion = progress->getPropInt("@format");
  255. connected = true;
  256. }
  257. void lockWrite()
  258. {
  259. if (connectedWrite) return;
  260. if (!rootPath.length())
  261. throw MakeStringException(WUERR_GraphProgressWriteUnsupported, "Writing to graph progress unsupported in this context");
  262. // JCSMORE - look at using changeMode here.
  263. if (conn)
  264. clearConnection();
  265. else
  266. packProgress(wuid,false);
  267. conn.setown(querySDS().connect(rootPath.str(), myProcessSession(), RTM_LOCK_WRITE|RTM_CREATE_QUERY, SDS_LOCK_TIMEOUT));
  268. progress.set(conn->queryRoot());
  269. if (!progress->hasChildren()) // i.e. blank.
  270. {
  271. formatVersion = PROGRESS_FORMAT_V;
  272. progress->setPropInt("@format", PROGRESS_FORMAT_V);
  273. }
  274. else
  275. formatVersion = progress->getPropInt("@format");
  276. connected = connectedWrite = true;
  277. }
  278. void unlock()
  279. {
  280. connected = false;
  281. connectedWrite = false;
  282. clearConnection();
  283. }
  284. static bool getRunningGraph(const char *wuid, IStringVal &graphName, WUGraphIDType &subId)
  285. {
  286. StringBuffer path;
  287. Owned<IRemoteConnection> conn = querySDS().connect(path.append("/GraphProgress/").append(wuid).str(), myProcessSession(), RTM_LOCK_READ, SDS_LOCK_TIMEOUT);
  288. if (!conn) return false;
  289. const char *name = conn->queryRoot()->queryProp("Running/@graph");
  290. if (name)
  291. {
  292. graphName.set(name);
  293. subId = conn->queryRoot()->getPropInt64("Running/@subId");
  294. return true;
  295. }
  296. else
  297. return false;
  298. }
  299. void setGraphState(WUGraphState state)
  300. {
  301. progress->setPropInt("@_state", (unsigned)state);
  302. }
  303. void setNodeState(WUGraphIDType nodeId, WUGraphState state)
  304. {
  305. if (!connectedWrite) lockWrite();
  306. StringBuffer path;
  307. path.append("node[@id=\"").append(nodeId).append("\"]");
  308. IPropertyTree *node = progress->queryPropTree(path.str());
  309. if (!node)
  310. {
  311. node = progress->addPropTree("node", createPTree());
  312. node->setPropInt("@id", (int)nodeId);
  313. }
  314. node->setPropInt("@_state", (unsigned)state);
  315. switch (state)
  316. {
  317. case WUGraphRunning:
  318. {
  319. StringBuffer path;
  320. Owned<IRemoteConnection> conn = querySDS().connect(path.append("/GraphProgress/").append(wuid).str(), myProcessSession(), RTM_LOCK_WRITE|RTM_CREATE_QUERY, SDS_LOCK_TIMEOUT);
  321. IPropertyTree *running = conn->queryRoot()->setPropTree("Running", createPTree());
  322. running->setProp("@graph", graphName);
  323. running->setPropInt64("@subId", nodeId);
  324. break;
  325. }
  326. case WUGraphComplete:
  327. {
  328. StringBuffer path;
  329. Owned<IRemoteConnection> conn = querySDS().connect(path.append("/GraphProgress/").append(wuid).str(), myProcessSession(), RTM_LOCK_WRITE, SDS_LOCK_TIMEOUT);
  330. conn->queryRoot()->removeProp("Running"); // only one thing running at any given time and one thing with lockWrite access
  331. break;
  332. }
  333. }
  334. }
  335. virtual IPropertyTree * getProgressTree()
  336. {
  337. if (!connected) connect();
  338. if (progress->getPropBool("@stats"))
  339. return createProcessTreeFromStats();
  340. return LINK(progress);
  341. }
  342. virtual WUGraphState queryGraphState()
  343. {
  344. return (WUGraphState)queryProgressStateTree()->getPropInt("@_state", (unsigned)WUGraphUnknown);
  345. }
  346. virtual WUGraphState queryNodeState(WUGraphIDType nodeId)
  347. {
  348. StringBuffer path;
  349. path.append("node[@id=\"").append(nodeId).append("\"]/@_state");
  350. return (WUGraphState)queryProgressStateTree()->getPropInt(path.str(), (unsigned)WUGraphUnknown);
  351. }
  352. virtual IWUGraphProgress * update()
  353. {
  354. return new CGraphProgress(*this);
  355. }
  356. virtual IWUGraphStats * update(StatisticCreatorType creatorType, const char * creator, unsigned subgraph)
  357. {
  358. return new CWuGraphStats(*this, creatorType, creator, graphName, subgraph);
  359. }
  360. virtual unsigned queryFormatVersion()
  361. {
  362. if (!connected) connect();
  363. return formatVersion;
  364. }
  365. static bool packProgress(const char *wuid,bool pack)
  366. {
  367. //MORE: This function could be deleted, because progress information is never actually packed
  368. StringBuffer path;
  369. path.append("/GraphProgress/").append(wuid);
  370. Owned<IRemoteConnection> conn(querySDS().connect(path.str(), myProcessSession(), RTM_LOCK_WRITE|RTM_CREATE_QUERY, SDS_LOCK_TIMEOUT));
  371. if (!conn)
  372. return false;
  373. Owned<IPropertyTree> newt;
  374. MemoryBuffer buf;
  375. IPropertyTree *root = conn->queryRoot();
  376. if (root->getPropBin("Packed",buf)) {
  377. if (pack)
  378. return true;
  379. newt.setown(createPTree(buf));
  380. IPropertyTree *running = root->queryPropTree("Running");
  381. if (running)
  382. newt->setPropTree("Running",createPTreeFromIPT(running));
  383. }
  384. else {
  385. if (!pack)
  386. return true;
  387. newt.setown(createPTree(wuid));
  388. IPropertyTree *running = root->queryPropTree("Running");
  389. if (running) {
  390. newt->setPropTree("Running",createPTreeFromIPT(running));
  391. root->removeTree(running);
  392. }
  393. root->serialize(buf);
  394. newt->setPropBin("Packed",buf.length(),buf.bufferBase());
  395. }
  396. root->setPropTree(NULL,newt.getClear());
  397. return true;
  398. }
  399. private:
  400. IPropertyTree * queryProgressStateTree()
  401. {
  402. if (!connected) connect();
  403. return progress;
  404. }
  405. static void expandStats(IPropertyTree * target, IStatisticCollection & collection)
  406. {
  407. StringBuffer formattedValue;
  408. unsigned numStats = collection.getNumStatistics();
  409. for (unsigned i=0; i < numStats; i++)
  410. {
  411. StatisticKind kind;
  412. unsigned __int64 value;
  413. collection.getStatistic(kind, value, i);
  414. formatStatistic(formattedValue.clear(), value, kind);
  415. //Until 6.0 generate the backward compatible tag name
  416. const char * legacyTreeTag = queryLegacyTreeTag(kind);
  417. if (legacyTreeTag)
  418. {
  419. StatisticMeasure measure = queryMeasure(kind);
  420. if (measure == SMeasureSkew)
  421. {
  422. //Minimum stats were always output as +ve numbers
  423. if (queryStatsVariant(kind) == StSkewMin)
  424. value = -value;
  425. target->setPropInt64(legacyTreeTag, value/100);
  426. }
  427. else if (measure == SMeasureTimeNs)
  428. {
  429. //Legacy timings are in ms => scale
  430. target->setPropInt64(legacyTreeTag, value/1000000);
  431. }
  432. else
  433. target->setProp(legacyTreeTag, formattedValue);
  434. }
  435. //Unconditionally output in the new format.
  436. target->setProp(queryTreeTag(kind), formattedValue);
  437. }
  438. }
  439. void expandProcessTreeFromStats(IPropertyTree * rootTarget, IPropertyTree * target, IStatisticCollection * collection)
  440. {
  441. expandStats(target, *collection);
  442. StringBuffer scopeName;
  443. Owned<IStatisticCollectionIterator> activityIter = &collection->getScopes(NULL);
  444. ForEach(*activityIter)
  445. {
  446. IStatisticCollection & cur = activityIter->query();
  447. cur.getScope(scopeName.clear());
  448. const char * id = scopeName.str();
  449. const char * tag;
  450. IPropertyTree * curTarget = target;
  451. switch (cur.queryScopeType())
  452. {
  453. case SSTedge:
  454. tag = "edge";
  455. id += strlen(EdgeScopePrefix);
  456. break;
  457. case SSTactivity:
  458. tag = "node";
  459. id += strlen(ActivityScopePrefix);
  460. break;
  461. case SSTsubgraph:
  462. //All subgraphs are added a root elements in the progress tree
  463. curTarget = rootTarget;
  464. tag = "node";
  465. id += strlen(SubGraphScopePrefix);
  466. break;
  467. default:
  468. throwUnexpected();
  469. }
  470. IPropertyTree * next = curTarget->addPropTree(tag, createPTree());
  471. next->setProp("@id", id);
  472. expandProcessTreeFromStats(rootTarget, next, &cur);
  473. }
  474. }
  475. IPropertyTree * createProcessTreeFromStats()
  476. {
  477. MemoryBuffer compressed;
  478. MemoryBuffer serialized;
  479. Owned<IPropertyTree> progressTree = createPTree();
  480. Owned<IPropertyTreeIterator> iter = progress->getElements("sg*");
  481. ForEach(*iter)
  482. {
  483. IPropertyTree & curSubGraph = iter->query();
  484. curSubGraph.getPropBin("Stats", compressed.clear());
  485. //Protect against updates that delete the stats while we are iterating
  486. if (compressed.length())
  487. {
  488. decompressToBuffer(serialized.clear(), compressed);
  489. Owned<IStatisticCollection> collection = createStatisticCollection(serialized);
  490. expandProcessTreeFromStats(progressTree, progressTree, collection);
  491. }
  492. }
  493. return progressTree.getClear();
  494. }
  495. void setSubgraphStats(StatisticCreatorType creatorType, const char * creator, unsigned id, const MemoryBuffer & compressed, unsigned minActivity, unsigned maxActivity)
  496. {
  497. StringBuffer tag;
  498. tag.append("sg").append(id);
  499. IPropertyTree * subgraph = createPTree(tag);
  500. subgraph->setProp("@c", queryCreatorTypeName(creatorType));
  501. subgraph->setProp("@creator", creator);
  502. subgraph->setPropInt("@minActivity", minActivity);
  503. subgraph->setPropInt("@maxActivity", maxActivity);
  504. //Replace the particular subgraph statistics added by this creator
  505. tag.append("[@creator='").append(creator).append("']");
  506. lockWrite();
  507. subgraph = progress->setPropTree(tag, subgraph);
  508. subgraph->setPropBin("Stats", compressed.length(), compressed.toByteArray());
  509. if (!progress->getPropBool("@stats", false))
  510. progress->setPropBool("@stats", true);
  511. unlock();
  512. }
  513. private:
  514. Owned<IRemoteConnection> conn;
  515. Linked<IPropertyTree> progress;
  516. StringAttr wuid, graphName;
  517. StringBuffer rootPath;
  518. bool connected, connectedWrite;
  519. unsigned formatVersion;
  520. };
  521. //--------------------------------------------------------------------------------------------------------------------
  522. class ExtractedStatistic : public CInterfaceOf<IConstWUStatistic>
  523. {
  524. public:
  525. virtual IStringVal & getDescription(IStringVal & str, bool createDefault) const
  526. {
  527. if (!description && createDefault)
  528. {
  529. switch (kind)
  530. {
  531. case StTimeElapsed:
  532. {
  533. if (scopeType != SSTsubgraph)
  534. break;
  535. //Create a default description for a root subgraph
  536. const char * colon = strchr(scope, ':');
  537. if (!colon)
  538. break;
  539. const char * subgraph = colon+1;
  540. //Check for nested subgraph
  541. if (strchr(subgraph, ':'))
  542. break;
  543. assertex(strncmp(subgraph, SubGraphScopePrefix, strlen(SubGraphScopePrefix)) == 0);
  544. StringAttr graphname;
  545. graphname.set(scope, colon - scope);
  546. unsigned subId = atoi(subgraph + strlen(SubGraphScopePrefix));
  547. StringBuffer desc;
  548. formatGraphTimerLabel(desc, graphname, 0, subId);
  549. str.set(desc);
  550. return str;
  551. }
  552. }
  553. }
  554. str.set(description);
  555. return str;
  556. }
  557. virtual IStringVal & getCreator(IStringVal & str) const
  558. {
  559. str.set(creator);
  560. return str;
  561. }
  562. virtual IStringVal & getScope(IStringVal & str) const
  563. {
  564. str.set(scope);
  565. return str;
  566. }
  567. virtual IStringVal & getFormattedValue(IStringVal & str) const
  568. {
  569. StringBuffer formatted;
  570. formatStatistic(formatted, value, measure);
  571. str.set(formatted);
  572. return str;
  573. }
  574. virtual StatisticMeasure getMeasure() const
  575. {
  576. return measure;
  577. }
  578. virtual StatisticKind getKind() const
  579. {
  580. return kind;
  581. }
  582. virtual StatisticCreatorType getCreatorType() const
  583. {
  584. return creatorType;
  585. }
  586. virtual StatisticScopeType getScopeType() const
  587. {
  588. return scopeType;
  589. }
  590. virtual unsigned __int64 getValue() const
  591. {
  592. return value;
  593. }
  594. virtual unsigned __int64 getCount() const
  595. {
  596. return count;
  597. }
  598. virtual unsigned __int64 getMax() const
  599. {
  600. return max;
  601. }
  602. virtual unsigned __int64 getTimestamp() const
  603. {
  604. return timeStamp;
  605. }
  606. virtual bool matches(const IStatisticsFilter * filter) const
  607. {
  608. return filter->matches(creatorType, creator, scopeType, scope, measure, kind);
  609. }
  610. public:
  611. StringAttr creator;
  612. StringAttr description;
  613. StringBuffer scope;
  614. StatisticMeasure measure;
  615. StatisticKind kind;
  616. StatisticCreatorType creatorType;
  617. StatisticScopeType scopeType;
  618. unsigned __int64 value;
  619. unsigned __int64 count;
  620. unsigned __int64 max;
  621. unsigned __int64 timeStamp;
  622. };
  623. class CConstGraphProgressStatisticsIterator : public CInterfaceOf<IConstWUStatisticIterator>
  624. {
  625. public:
  626. CConstGraphProgressStatisticsIterator(const char * wuid, const IStatisticsFilter * _filter) : filter(_filter)
  627. {
  628. if (filter)
  629. scopes.appendList(filter->queryScope(), ":");
  630. const char * searchGraph = "*";
  631. if (scopes.ordinality())
  632. searchGraph = scopes.item(0);
  633. rootPath.append("/GraphProgress/").append(wuid).append('/');
  634. bool singleGraph = false;
  635. if (!containsWildcard(searchGraph))
  636. {
  637. rootPath.append(searchGraph).append("/");
  638. singleGraph = true;
  639. }
  640. //Don't lock the statistics while we iterate - any partial updates must not cause problems
  641. if (daliClientActive())
  642. conn.setown(querySDS().connect(rootPath.str(), myProcessSession(), RTM_NONE, SDS_LOCK_TIMEOUT));
  643. if (conn && !singleGraph)
  644. graphIter.setown(conn->queryRoot()->getElements("*"));
  645. curStat.setown(new ExtractedStatistic);
  646. //These are currently constant for all graph statistics instances
  647. curStat->count = 1;
  648. curStat->max = 0;
  649. valid = false;
  650. }
  651. virtual IConstWUStatistic & query()
  652. {
  653. return *curStat;
  654. }
  655. virtual bool first()
  656. {
  657. valid = false;
  658. if (!conn)
  659. return false;
  660. if (graphIter && !graphIter->first())
  661. return false;
  662. ensureUniqueStatistic();
  663. if (!firstSubGraph())
  664. return false;
  665. valid = true;
  666. return true;
  667. }
  668. virtual bool next()
  669. {
  670. ensureUniqueStatistic();
  671. if (!nextStatistic())
  672. {
  673. if (!nextSubGraph())
  674. {
  675. if (!nextGraph())
  676. {
  677. valid = false;
  678. return false;
  679. }
  680. }
  681. }
  682. return true;
  683. }
  684. virtual bool isValid()
  685. {
  686. return valid;
  687. }
  688. protected:
  689. bool firstSubGraph()
  690. {
  691. IPropertyTree & graphNode = graphIter ? graphIter->query() : *conn->queryRoot();
  692. const char * xpath = "sg*";
  693. StringBuffer childXpath;
  694. if (scopes.isItem(1))
  695. {
  696. const char * scope1 = scopes.item(1);
  697. if (strnicmp(scope1, "sg", 2) == 0)
  698. {
  699. childXpath.append(scope1);
  700. xpath = childXpath.str();
  701. }
  702. }
  703. subgraphIter.setown(graphNode.getElements(xpath));
  704. if (!subgraphIter)
  705. subgraphIter.setown(graphNode.getElements("sg0"));
  706. if (!subgraphIter->first())
  707. return false;
  708. if (firstStat())
  709. return true;
  710. return nextSubGraph();
  711. }
  712. bool nextSubGraph()
  713. {
  714. loop
  715. {
  716. if (!subgraphIter->next())
  717. return false;
  718. if (firstStat())
  719. return true;
  720. }
  721. }
  722. bool nextGraph()
  723. {
  724. if (!graphIter)
  725. return false;
  726. loop
  727. {
  728. if (!graphIter->next())
  729. return false;
  730. if (firstSubGraph())
  731. return true;
  732. }
  733. }
  734. bool firstStat()
  735. {
  736. IPropertyTree & curSubGraph = subgraphIter->query();
  737. if (!checkSubGraph())
  738. return false;
  739. curSubGraph.getPropBin("Stats", compressed.clear());
  740. //Don't crash on old format progress...
  741. if (compressed.length() == 0)
  742. return false;
  743. decompressToBuffer(serialized.clear(), compressed);
  744. Owned<IStatisticCollection> collection = createStatisticCollection(serialized);
  745. curStat->timeStamp = collection->queryWhenCreated();
  746. return beginCollection(*collection);
  747. }
  748. bool beginCollection(IStatisticCollection & collection)
  749. {
  750. collections.append(OLINK(collection));
  751. numStats = collection.getNumStatistics();
  752. curStatIndex = 0;
  753. if (checkScope())
  754. {
  755. if (curStatIndex < numStats)
  756. {
  757. if (checkStatistic())
  758. return true;
  759. return nextStatistic();
  760. }
  761. }
  762. return nextChildScope();
  763. }
  764. bool nextStatistic()
  765. {
  766. //Finish iterating the statistics at this level.
  767. while (++curStatIndex < numStats)
  768. {
  769. if (checkStatistic())
  770. return true;
  771. }
  772. return nextChildScope();
  773. }
  774. bool nextChildScope()
  775. {
  776. loop
  777. {
  778. if (collections.ordinality() == 0)
  779. return false;
  780. IStatisticCollection * curCollection = &collections.tos();
  781. if (childIterators.ordinality() < collections.ordinality())
  782. {
  783. if (!filter || filter->recurseChildScopes(curStat->scopeType, curStat->scope))
  784. {
  785. //Start iterating the children for the current collection
  786. childIterators.append(curCollection->getScopes(NULL));
  787. if (!childIterators.tos().first())
  788. {
  789. finishCollection();
  790. continue;
  791. }
  792. }
  793. else
  794. {
  795. //Don't walk the child scopes
  796. collections.pop();
  797. continue;
  798. }
  799. }
  800. else if (!childIterators.tos().next())
  801. {
  802. finishCollection();
  803. continue;
  804. }
  805. if (beginCollection(childIterators.tos().query()))
  806. return true;
  807. }
  808. }
  809. void finishCollection()
  810. {
  811. collections.pop();
  812. childIterators.pop();
  813. }
  814. bool checkSubGraph()
  815. {
  816. if (!filter)
  817. return true;
  818. IPropertyTree & curSubGraph = subgraphIter->query();
  819. curStat->creatorType = queryCreatorType(curSubGraph.queryProp("@c"));
  820. curStat->creator.set(curSubGraph.queryProp("@creator"));
  821. return filter->matches(curStat->creatorType, curStat->creator, SSTall, NULL, SMeasureAll, StKindAll);
  822. }
  823. bool checkScope()
  824. {
  825. if (!filter)
  826. return true;
  827. IStatisticCollection * collection = &collections.tos();
  828. curStat->scopeType = collection->queryScopeType();
  829. collection->getFullScope(curStat->scope.clear());
  830. return filter->matches(SCTall, NULL, curStat->scopeType, curStat->scope, SMeasureAll, StKindAll);
  831. }
  832. bool checkStatistic()
  833. {
  834. IStatisticCollection & collection = collections.tos();
  835. collection.getStatistic(curStat->kind, curStat->value, curStatIndex);
  836. curStat->measure = queryMeasure(curStat->kind);
  837. if (!filter)
  838. return true;
  839. if (!filter->matches(SCTall, NULL, SSTall, NULL, curStat->measure, curStat->kind))
  840. return false;
  841. return true;
  842. }
  843. void ensureUniqueStatistic()
  844. {
  845. //If something else has linked this statistic, clone a unique one.
  846. if (curStat->IsShared())
  847. curStat.setown(new ExtractedStatistic(*curStat));
  848. }
  849. private:
  850. Owned<IRemoteConnection> conn;
  851. Owned<ExtractedStatistic> curStat;
  852. const IStatisticsFilter * filter;
  853. StringArray scopes;
  854. StringBuffer rootPath;
  855. Owned<IPropertyTreeIterator> graphIter;
  856. Owned<IPropertyTreeIterator> subgraphIter;
  857. IArrayOf<IStatisticCollection> collections;
  858. IArrayOf<IStatisticCollectionIterator> childIterators;
  859. MemoryBuffer compressed;
  860. MemoryBuffer serialized;
  861. unsigned numStats;
  862. unsigned curStatIndex;
  863. bool valid;
  864. };
  865. //--------------------------------------------------------------------------------------------------------------------
  866. struct mapEnums { int val; const char *str; };
  867. mapEnums states[] = {
  868. { WUStateUnknown, "unknown" },
  869. { WUStateCompiled, "compiled" },
  870. { WUStateRunning, "running" },
  871. { WUStateCompleted, "completed" },
  872. { WUStateFailed, "failed" },
  873. { WUStateArchived, "archived" },
  874. { WUStateAborting, "aborting" },
  875. { WUStateAborted, "aborted" },
  876. { WUStateBlocked, "blocked" },
  877. { WUStateSubmitted, "submitted" },
  878. { WUStateScheduled, "scheduled" },
  879. { WUStateCompiling, "compiling" },
  880. { WUStateWait, "wait" },
  881. { WUStateUploadingFiles, "uploading_files" },
  882. { WUStateDebugPaused, "debugging" },
  883. { WUStateDebugRunning, "debug_running" },
  884. { WUStatePaused, "paused" },
  885. { WUStateSize, NULL }
  886. };
  887. const char * getWorkunitStateStr(WUState state)
  888. {
  889. dbgassertex(state < WUStateSize);
  890. return states[state].str; // MORE - should be using getEnumText, or need to take steps to ensure values remain contiguous and in order.
  891. }
  892. const char *getEnumText(int value, const mapEnums *map)
  893. {
  894. const char *defval = map->str;
  895. while (map->str)
  896. {
  897. if (value==map->val)
  898. return map->str;
  899. map++;
  900. }
  901. assertex(!"Unexpected value in setEnum");
  902. return defval;
  903. }
  904. void setEnum(IPropertyTree *p, const char *propname, int value, const mapEnums *map)
  905. {
  906. const char *defval = map->str;
  907. while (map->str)
  908. {
  909. if (value==map->val)
  910. {
  911. p->setProp(propname, map->str);
  912. return;
  913. }
  914. map++;
  915. }
  916. assertex(!"Unexpected value in setEnum");
  917. p->setProp(propname, defval);
  918. }
  919. static int getEnum(const char *v, const mapEnums *map)
  920. {
  921. if (v)
  922. {
  923. while (map->str)
  924. {
  925. if (stricmp(v, map->str)==0)
  926. return map->val;
  927. map++;
  928. }
  929. assertex(!"Unexpected value in getEnum");
  930. }
  931. return 0;
  932. }
  933. static int getEnum(const IPropertyTree *p, const char *propname, const mapEnums *map)
  934. {
  935. return getEnum(p->queryProp(propname),map);
  936. }
  937. //==========================================================================================
  938. class CLightweightWorkunitInfo : public CInterfaceOf<IConstWorkUnitInfo>
  939. {
  940. public:
  941. CLightweightWorkunitInfo(IPropertyTree &p)
  942. {
  943. wuid.set(p.queryName());
  944. user.set(p.queryProp("@submitID"));
  945. jobName.set(p.queryProp("@jobName"));
  946. clusterName.set(p.queryProp("@clusterName"));
  947. timeScheduled.set(p.queryProp("@timeScheduled"));
  948. state = (WUState) getEnum(&p, "@state", states);
  949. wuscope.set(p.queryProp("@scope"));
  950. _isProtected = p.getPropBool("@protected", false);
  951. _isAborting = false; // MORE - this one is tricky
  952. }
  953. virtual const char *queryWuid() const { return wuid.str(); }
  954. virtual const char *queryUser() const { return user.str(); }
  955. virtual const char *queryJobName() const { return jobName.str(); }
  956. virtual const char *queryClusterName() const { return clusterName.str(); }
  957. virtual const char *queryWuScope() const { return wuscope.str(); }
  958. virtual WUState getState() const { return state; }
  959. virtual const char *queryStateDesc() const { return getEnumText(getState(), states); }
  960. virtual bool isProtected() const { return _isProtected; }
  961. // Not sure about these ones...
  962. virtual bool aborting() const { return _isAborting; }
  963. virtual IJlibDateTime & getTimeScheduled(IJlibDateTime & val) const
  964. {
  965. if (timeScheduled.length())
  966. val.setGmtString(timeScheduled.str());
  967. return val;
  968. }
  969. protected:
  970. StringAttr wuid, user, jobName, clusterName, timeScheduled, wuscope;
  971. WUState state;
  972. bool _isProtected;
  973. bool _isAborting;
  974. };
  975. extern IConstWorkUnitInfo *createConstWorkUnitInfo(IPropertyTree &p)
  976. {
  977. return new CLightweightWorkunitInfo(p);
  978. }
  979. class CDaliWorkUnit : public CLocalWorkUnit
  980. {
  981. public:
  982. IMPLEMENT_IINTERFACE;
  983. CDaliWorkUnit(IRemoteConnection *_conn, ISecManager *secmgr, ISecUser *secuser)
  984. : connection(_conn), CLocalWorkUnit(secmgr, secuser)
  985. {
  986. loadPTree(connection->getRoot());
  987. abortDirty = true;
  988. abortState = false;
  989. }
  990. ~CDaliWorkUnit()
  991. {
  992. // NOTE - order is important - we need to construct connection before p and (especially) destroy after p
  993. // We use the beforeDIspose() in base class to help ensure this
  994. p.clear();
  995. }
  996. virtual void forceReload()
  997. {
  998. synchronized sync(locked); // protect locked workunits (uncommited writes) from reload
  999. StringBuffer wuRoot;
  1000. getXPath(wuRoot, p->queryName());
  1001. IRemoteConnection *newconn = querySDS().connect(wuRoot.str(), myProcessSession(), 0, SDS_LOCK_TIMEOUT);
  1002. if (!newconn)
  1003. throw MakeStringException(WUERR_ConnectFailed, "Could not connect to workunit %s (deleted?)",p->queryName());
  1004. CriticalBlock block(crit);
  1005. clearCached(true);
  1006. connection.setown(newconn);
  1007. abortDirty = true;
  1008. p.setown(connection->getRoot());
  1009. }
  1010. virtual void cleanupAndDelete(bool deldll, bool deleteOwned, const StringArray *deleteExclusions)
  1011. {
  1012. CLocalWorkUnit::cleanupAndDelete(deldll, deleteOwned, deleteExclusions);
  1013. connection->close(true);
  1014. connection.clear();
  1015. }
  1016. virtual void commit()
  1017. {
  1018. CLocalWorkUnit::commit();
  1019. if (connection)
  1020. connection->commit();
  1021. }
  1022. virtual void _lockRemote()
  1023. {
  1024. StringBuffer wuRoot;
  1025. getXPath(wuRoot, p->queryName());
  1026. if (connection)
  1027. connection->changeMode(RTM_LOCK_WRITE,SDS_LOCK_TIMEOUT);
  1028. else
  1029. connection.setown(querySDS().connect(wuRoot.str(), myProcessSession(), RTM_LOCK_WRITE, SDS_LOCK_TIMEOUT));
  1030. if (!connection)
  1031. throw MakeStringException(WUERR_LockFailed, "Failed to get connection for xpath %s", wuRoot.str());
  1032. clearCached(true);
  1033. abortDirty = true;
  1034. p.setown(connection->getRoot());
  1035. }
  1036. virtual void _unlockRemote()
  1037. {
  1038. try
  1039. {
  1040. //MORE: I'm not convinced this is useful...
  1041. setStatistic(queryStatisticsComponentType(), queryStatisticsComponentName(), SSTglobal, NULL, StWhenWorkunitModified, NULL, getTimeStampNowValue(), 1, 0, StatsMergeReplace);
  1042. try
  1043. {
  1044. connection->commit();
  1045. }
  1046. catch (IException *e)
  1047. {
  1048. EXCLOG(e, "Error during workunit commit");
  1049. connection->rollback();
  1050. connection->changeMode(0, SDS_LOCK_TIMEOUT);
  1051. throw;
  1052. }
  1053. connection->changeMode(0, SDS_LOCK_TIMEOUT);
  1054. }
  1055. catch (IException *E)
  1056. {
  1057. StringBuffer s;
  1058. PrintLog("Failed to release write lock on workunit: %s", E->errorMessage(s).str());
  1059. throw;
  1060. }
  1061. }
  1062. virtual void subscribe(WUSubscribeOptions options)
  1063. {
  1064. CriticalBlock block(crit);
  1065. assertex(options==SubscribeOptionAbort);
  1066. if (!abortWatcher)
  1067. {
  1068. abortWatcher.setown(new CWorkUnitAbortWatcher(this, p->queryName()));
  1069. abortDirty = true;
  1070. }
  1071. }
  1072. virtual void unsubscribe()
  1073. {
  1074. CriticalBlock block(crit);
  1075. if (abortWatcher)
  1076. {
  1077. abortWatcher->unsubscribe();
  1078. abortWatcher.clear();
  1079. }
  1080. }
  1081. virtual bool aborting() const
  1082. {
  1083. CriticalBlock block(crit);
  1084. if (abortDirty)
  1085. {
  1086. StringBuffer apath;
  1087. apath.append("/WorkUnitAborts/").append(p->queryName());
  1088. Owned<IRemoteConnection> acon = querySDS().connect(apath.str(), myProcessSession(), 0, SDS_LOCK_TIMEOUT);
  1089. if (acon)
  1090. abortState = acon->queryRoot()->getPropInt(NULL) != 0;
  1091. else
  1092. abortState = false;
  1093. abortDirty = false;
  1094. }
  1095. return abortState;
  1096. }
  1097. protected:
  1098. class CWorkUnitAbortWatcher : public CInterface, implements ISDSSubscription
  1099. {
  1100. CDaliWorkUnit *parent; // not linked - it links me
  1101. SubscriptionId abort;
  1102. public:
  1103. IMPLEMENT_IINTERFACE;
  1104. CWorkUnitAbortWatcher(CDaliWorkUnit *_parent, const char *wuid) : parent(_parent)
  1105. {
  1106. StringBuffer wuRoot;
  1107. wuRoot.append("/WorkUnitAborts/").append(wuid);
  1108. abort = querySDS().subscribe(wuRoot.str(), *this);
  1109. }
  1110. ~CWorkUnitAbortWatcher()
  1111. {
  1112. assertex(abort==0);
  1113. }
  1114. void unsubscribe()
  1115. {
  1116. querySDS().unsubscribe(abort);
  1117. abort = 0;
  1118. }
  1119. void notify(SubscriptionId id, const char *xpath, SDSNotifyFlags flags, unsigned valueLen, const void *valueData)
  1120. {
  1121. parent->abortDirty = true;
  1122. }
  1123. };
  1124. Owned<IRemoteConnection> connection;
  1125. Owned<CWorkUnitAbortWatcher> abortWatcher;
  1126. mutable bool abortDirty;
  1127. mutable bool abortState;
  1128. };
  1129. class CLockedWorkUnit : public CInterface, implements ILocalWorkUnit, implements IExtendedWUInterface
  1130. {
  1131. public:
  1132. Owned<CLocalWorkUnit> c;
  1133. IMPLEMENT_IINTERFACE;
  1134. CLockedWorkUnit(CLocalWorkUnit *_c) : c(_c) {}
  1135. ~CLockedWorkUnit()
  1136. {
  1137. if (workUnitTraceLevel > 1)
  1138. PrintLog("Releasing locked workunit %s", queryWuid());
  1139. if (c)
  1140. c->unlockRemote();
  1141. }
  1142. virtual IConstWorkUnit * unlock()
  1143. {
  1144. c->unlockRemote();
  1145. return c.getClear();
  1146. }
  1147. virtual bool aborting() const
  1148. { return c->aborting(); }
  1149. virtual void forceReload()
  1150. { UNIMPLEMENTED; }
  1151. virtual WUAction getAction() const
  1152. { return c->getAction(); }
  1153. virtual IStringVal& getActionEx(IStringVal & str) const
  1154. { return c->getActionEx(str); }
  1155. virtual IStringVal & getApplicationValue(const char * application, const char * propname, IStringVal & str) const
  1156. { return c->getApplicationValue(application, propname, str); }
  1157. virtual int getApplicationValueInt(const char * application, const char * propname, int defVal) const
  1158. { return c->getApplicationValueInt(application, propname, defVal); }
  1159. virtual IConstWUAppValueIterator & getApplicationValues() const
  1160. { return c->getApplicationValues(); }
  1161. virtual bool hasWorkflow() const
  1162. { return c->hasWorkflow(); }
  1163. virtual unsigned queryEventScheduledCount() const
  1164. { return c->queryEventScheduledCount(); }
  1165. virtual IPropertyTree * queryWorkflowTree() const
  1166. { return c->queryWorkflowTree(); }
  1167. virtual IConstWorkflowItemIterator * getWorkflowItems() const
  1168. { return c->getWorkflowItems(); }
  1169. virtual IWorkflowItemArray * getWorkflowClone() const
  1170. { return c->getWorkflowClone(); }
  1171. virtual bool requiresLocalFileUpload() const
  1172. { return c->requiresLocalFileUpload(); }
  1173. virtual IConstLocalFileUploadIterator * getLocalFileUploads() const
  1174. { return c->getLocalFileUploads(); }
  1175. virtual bool getIsQueryService() const
  1176. { return c->getIsQueryService(); }
  1177. virtual bool getCloneable() const
  1178. { return c->getCloneable(); }
  1179. virtual IUserDescriptor * queryUserDescriptor() const
  1180. { return c->queryUserDescriptor(); }
  1181. virtual const char *queryClusterName() const
  1182. { return c->queryClusterName(); }
  1183. virtual unsigned getCodeVersion() const
  1184. { return c->getCodeVersion(); }
  1185. virtual unsigned getWuidVersion() const
  1186. { return c->getWuidVersion(); }
  1187. virtual void getBuildVersion(IStringVal & buildVersion, IStringVal & eclVersion) const
  1188. { c->getBuildVersion(buildVersion, eclVersion); }
  1189. virtual bool hasDebugValue(const char * propname) const
  1190. { return c->hasDebugValue(propname); }
  1191. virtual IStringVal & getDebugValue(const char * propname, IStringVal & str) const
  1192. { return c->getDebugValue(propname, str); }
  1193. virtual int getDebugValueInt(const char * propname, int defVal) const
  1194. { return c->getDebugValueInt(propname, defVal); }
  1195. virtual __int64 getDebugValueInt64(const char * propname, __int64 defVal) const
  1196. { return c->getDebugValueInt64(propname, defVal); }
  1197. virtual bool getDebugValueBool(const char * propname, bool defVal) const
  1198. { return c->getDebugValueBool(propname, defVal); }
  1199. virtual IStringIterator & getDebugValues() const
  1200. { return c->getDebugValues(NULL); }
  1201. virtual IStringIterator & getDebugValues(const char *prop) const
  1202. { return c->getDebugValues(prop); }
  1203. virtual unsigned getExceptionCount() const
  1204. { return c->getExceptionCount(); }
  1205. virtual IConstWUExceptionIterator & getExceptions() const
  1206. { return c->getExceptions(); }
  1207. virtual unsigned getGraphCount() const
  1208. { return c->getGraphCount(); }
  1209. virtual unsigned getSourceFileCount() const
  1210. { return c->getSourceFileCount(); }
  1211. virtual unsigned getResultCount() const
  1212. { return c->getResultCount(); }
  1213. virtual unsigned getVariableCount() const
  1214. { return c->getVariableCount(); }
  1215. virtual unsigned getApplicationValueCount() const
  1216. { return c->getApplicationValueCount(); }
  1217. virtual IConstWUGraphIterator & getGraphs(WUGraphType type) const
  1218. { return c->getGraphs(type); }
  1219. virtual IConstWUGraphMetaIterator & getGraphsMeta(WUGraphType type) const
  1220. { return c->getGraphsMeta(type); }
  1221. virtual IConstWUGraph * getGraph(const char *name) const
  1222. { return c->getGraph(name); }
  1223. virtual IConstWUGraphProgress * getGraphProgress(const char * name) const
  1224. { return c->getGraphProgress(name); }
  1225. virtual const char *queryJobName() const
  1226. { return c->queryJobName(); }
  1227. virtual IConstWUPlugin * getPluginByName(const char * name) const
  1228. { return c->getPluginByName(name); }
  1229. virtual IConstWUPluginIterator & getPlugins() const
  1230. { return c->getPlugins(); }
  1231. virtual IConstWULibrary* getLibraryByName(const char *name) const
  1232. { return c->getLibraryByName(name); }
  1233. virtual IConstWULibraryIterator & getLibraries() const
  1234. { return c->getLibraries(); }
  1235. virtual WUPriorityClass getPriority() const
  1236. { return c->getPriority(); }
  1237. virtual int getPriorityLevel() const
  1238. { return c->getPriorityLevel(); }
  1239. virtual int getPriorityValue() const
  1240. { return c->getPriorityValue(); }
  1241. virtual IConstWUQuery * getQuery() const
  1242. { return c->getQuery(); }
  1243. virtual IConstWUWebServicesInfo * getWebServicesInfo() const
  1244. { return c->getWebServicesInfo(); }
  1245. virtual IConstWURoxieQueryInfo* getRoxieQueryInfo() const
  1246. { return c->getRoxieQueryInfo(); }
  1247. virtual bool getRescheduleFlag() const
  1248. { return c->getRescheduleFlag(); }
  1249. virtual IConstWUResult * getResultByName(const char * name) const
  1250. { return c->getResultByName(name); }
  1251. virtual IConstWUResult * getResultBySequence(unsigned seq) const
  1252. { return c->getResultBySequence(seq); }
  1253. virtual unsigned getResultLimit() const
  1254. { return c->getResultLimit(); }
  1255. virtual IConstWUResultIterator & getResults() const
  1256. { return c->getResults(); }
  1257. virtual IStringVal & getScope(IStringVal & str) const
  1258. { return c->getScope(str); }
  1259. virtual IStringVal & getSecurityToken(IStringVal & str) const
  1260. { return c->getSecurityToken(str); }
  1261. virtual WUState getState() const
  1262. { return c->getState(); }
  1263. virtual IStringVal & getStateEx(IStringVal & str) const
  1264. { return c->getStateEx(str); }
  1265. virtual __int64 getAgentSession() const
  1266. { return c->getAgentSession(); }
  1267. virtual unsigned getAgentPID() const
  1268. { return c->getAgentPID(); }
  1269. virtual const char *queryStateDesc() const
  1270. { return c->queryStateDesc(); }
  1271. virtual bool getRunningGraph(IStringVal & graphName, WUGraphIDType & subId) const
  1272. { return c->getRunningGraph(graphName, subId); }
  1273. virtual IConstWUStatisticIterator & getStatistics(const IStatisticsFilter * filter) const
  1274. { return c->getStatistics(filter); }
  1275. virtual IConstWUStatistic * getStatistic(const char * creator, const char * scope, StatisticKind kind) const
  1276. { return c->getStatistic(creator, scope, kind); }
  1277. virtual IStringVal & getSnapshot(IStringVal & str) const
  1278. { return c->getSnapshot(str); }
  1279. virtual const char *queryUser() const
  1280. { return c->queryUser(); }
  1281. virtual ErrorSeverity getWarningSeverity(unsigned code, ErrorSeverity defaultSeverity) const
  1282. { return c->getWarningSeverity(code, defaultSeverity); }
  1283. virtual const char *queryWuScope() const
  1284. { return c->queryWuScope(); }
  1285. virtual const char *queryWuid() const
  1286. { return c->queryWuid(); }
  1287. virtual IConstWUResult * getGlobalByName(const char * name) const
  1288. { return c->getGlobalByName(name); }
  1289. virtual IConstWUResult * getTemporaryByName(const char * name) const
  1290. { return c->getTemporaryByName(name); }
  1291. virtual IConstWUResultIterator & getTemporaries() const
  1292. { return c->getTemporaries(); }
  1293. virtual IConstWUResult * getVariableByName(const char * name) const
  1294. { return c->getVariableByName(name); }
  1295. virtual IConstWUResultIterator & getVariables() const
  1296. { return c->getVariables(); }
  1297. virtual bool isProtected() const
  1298. { return c->isProtected(); }
  1299. virtual bool isPausing() const
  1300. { return c->isPausing(); }
  1301. virtual IWorkUnit & lock()
  1302. { ((CInterface *)this)->Link(); return (IWorkUnit &) *this; }
  1303. virtual bool reload()
  1304. { UNIMPLEMENTED; }
  1305. virtual void subscribe(WUSubscribeOptions options)
  1306. { c->subscribe(options); }
  1307. virtual void requestAbort()
  1308. { c->requestAbort(); }
  1309. virtual unsigned calculateHash(unsigned prevHash)
  1310. { return queryExtendedWU(c)->calculateHash(prevHash); }
  1311. virtual void copyWorkUnit(IConstWorkUnit *cached, bool all)
  1312. { queryExtendedWU(c)->copyWorkUnit(cached, all); }
  1313. virtual IPropertyTree *queryPTree() const
  1314. { return queryExtendedWU(c)->queryPTree(); }
  1315. virtual IPropertyTree *getUnpackedTree(bool includeProgress) const
  1316. { return queryExtendedWU(c)->getUnpackedTree(includeProgress); }
  1317. virtual bool archiveWorkUnit(const char *base,bool del,bool deldll,bool deleteOwned)
  1318. { return queryExtendedWU(c)->archiveWorkUnit(base,del,deldll,deleteOwned); }
  1319. virtual void packWorkUnit(bool pack)
  1320. { queryExtendedWU(c)->packWorkUnit(pack); }
  1321. virtual unsigned queryFileUsage(const char *filename) const
  1322. { return c->queryFileUsage(filename); }
  1323. virtual IJlibDateTime & getTimeScheduled(IJlibDateTime &val) const
  1324. { return c->getTimeScheduled(val); }
  1325. virtual unsigned getDebugAgentListenerPort() const
  1326. { return c->getDebugAgentListenerPort(); }
  1327. virtual IStringVal & getDebugAgentListenerIP(IStringVal &ip) const
  1328. { return c->getDebugAgentListenerIP(ip); }
  1329. virtual IStringVal & getXmlParams(IStringVal & params, bool hidePasswords) const
  1330. { return c->getXmlParams(params, hidePasswords); }
  1331. virtual const IPropertyTree *getXmlParams() const
  1332. { return c->getXmlParams(); }
  1333. virtual unsigned __int64 getHash() const
  1334. { return c->getHash(); }
  1335. virtual IStringIterator *getLogs(const char *type, const char *instance) const
  1336. { return c->getLogs(type, instance); }
  1337. virtual IStringIterator *getProcesses(const char *type) const
  1338. { return c->getProcesses(type); }
  1339. virtual IPropertyTreeIterator* getProcesses(const char *type, const char *instance) const
  1340. { return c->getProcesses(type, instance); }
  1341. virtual void clearExceptions()
  1342. { c->clearExceptions(); }
  1343. virtual void commit()
  1344. { c->commit(); }
  1345. virtual IWUException * createException()
  1346. { return c->createException(); }
  1347. virtual void addProcess(const char *type, const char *instance, unsigned pid, const char *log)
  1348. { c->addProcess(type, instance, pid, log); }
  1349. virtual void protect(bool protectMode)
  1350. { c->protect(protectMode); }
  1351. virtual void setAction(WUAction action)
  1352. { c->setAction(action); }
  1353. virtual void setApplicationValue(const char * application, const char * propname, const char * value, bool overwrite)
  1354. { c->setApplicationValue(application, propname, value, overwrite); }
  1355. virtual void setApplicationValueInt(const char * application, const char * propname, int value, bool overwrite)
  1356. { c->setApplicationValueInt(application, propname, value, overwrite); }
  1357. virtual void incEventScheduledCount()
  1358. { c->incEventScheduledCount(); }
  1359. virtual void setIsQueryService(bool value)
  1360. { c->setIsQueryService(value); }
  1361. virtual void setCloneable(bool value)
  1362. { c->setCloneable(value); }
  1363. virtual void setIsClone(bool value)
  1364. { c->setIsClone(value); }
  1365. virtual void setClusterName(const char * value)
  1366. { c->setClusterName(value); }
  1367. virtual void setCodeVersion(unsigned version, const char * buildVersion, const char * eclVersion)
  1368. { c->setCodeVersion(version, buildVersion, eclVersion); }
  1369. virtual void setDebugValue(const char * propname, const char * value, bool overwrite)
  1370. { c->setDebugValue(propname, value, overwrite); }
  1371. virtual void setDebugValueInt(const char * propname, int value, bool overwrite)
  1372. { c->setDebugValueInt(propname, value, overwrite); }
  1373. virtual void setJobName(const char * value)
  1374. { c->setJobName(value); }
  1375. virtual void setPriority(WUPriorityClass cls)
  1376. { c->setPriority(cls); }
  1377. virtual void setPriorityLevel(int level)
  1378. { c->setPriorityLevel(level); }
  1379. virtual void setRescheduleFlag(bool value)
  1380. { c->setRescheduleFlag(value); }
  1381. virtual void setResultLimit(unsigned value)
  1382. { c->setResultLimit(value); }
  1383. virtual void setSecurityToken(const char *value)
  1384. { c->setSecurityToken(value); }
  1385. virtual void setState(WUState state)
  1386. { c->setState(state); }
  1387. virtual void setStateEx(const char * text)
  1388. { c->setStateEx(text); }
  1389. virtual void setAgentSession(__int64 sessionId)
  1390. { c->setAgentSession(sessionId); }
  1391. virtual void setStatistic(StatisticCreatorType creatorType, const char * creator, StatisticScopeType scopeType, const char * scope, StatisticKind kind, const char * optDescription, unsigned __int64 value, unsigned __int64 count, unsigned __int64 maxValue, StatsMergeAction mergeAction)
  1392. { c->setStatistic(creatorType, creator, scopeType, scope, kind, optDescription, value, count, maxValue, mergeAction); }
  1393. virtual void setTracingValue(const char * propname, const char * value)
  1394. { c->setTracingValue(propname, value); }
  1395. virtual void setTracingValueInt(const char * propname, int value)
  1396. { c->setTracingValueInt(propname, value); }
  1397. virtual void setUser(const char * value)
  1398. { c->setUser(value); }
  1399. virtual void setWuScope(const char * value)
  1400. { c->setWuScope(value); }
  1401. virtual IWorkflowItem* addWorkflowItem(unsigned wfid, WFType type, WFMode mode, unsigned success, unsigned failure, unsigned recovery, unsigned retriesAllowed, unsigned contingencyFor)
  1402. { return c->addWorkflowItem(wfid, type, mode, success, failure, recovery, retriesAllowed, contingencyFor); }
  1403. virtual void syncRuntimeWorkflow(IWorkflowItemArray * array)
  1404. { c->syncRuntimeWorkflow(array); }
  1405. virtual IWorkflowItemIterator * updateWorkflowItems()
  1406. { return c->updateWorkflowItems(); }
  1407. virtual void resetWorkflow()
  1408. { c->resetWorkflow(); }
  1409. virtual void schedule()
  1410. { c->schedule(); }
  1411. virtual void deschedule()
  1412. { c->deschedule(); }
  1413. virtual unsigned addLocalFileUpload(LocalFileUploadType type, char const * source, char const * destination, char const * eventTag)
  1414. { return c->addLocalFileUpload(type, source, destination, eventTag); }
  1415. virtual IWUResult * updateGlobalByName(const char * name)
  1416. { return c->updateGlobalByName(name); }
  1417. virtual IWUGraph * createGraph(const char * name, WUGraphType type, IPropertyTree *xgmml)
  1418. { return c->createGraph(name, type, xgmml); }
  1419. virtual IWUGraph * updateGraph(const char * name)
  1420. { return c->updateGraph(name); }
  1421. virtual IWUQuery * updateQuery()
  1422. { return c->updateQuery(); }
  1423. virtual IWUWebServicesInfo * updateWebServicesInfo(bool create)
  1424. { return c->updateWebServicesInfo(create); }
  1425. virtual IWURoxieQueryInfo * updateRoxieQueryInfo(const char *wuid, const char *roxieClusterName)
  1426. { return c->updateRoxieQueryInfo(wuid, roxieClusterName); }
  1427. virtual IWUPlugin * updatePluginByName(const char * name)
  1428. { return c->updatePluginByName(name); }
  1429. virtual IWULibrary * updateLibraryByName(const char * name)
  1430. { return c->updateLibraryByName(name); }
  1431. virtual IWUResult * updateResultByName(const char * name)
  1432. { return c->updateResultByName(name); }
  1433. virtual IWUResult * updateResultBySequence(unsigned seq)
  1434. { return c->updateResultBySequence(seq); }
  1435. virtual IWUResult * updateTemporaryByName(const char * name)
  1436. { return c->updateTemporaryByName(name); }
  1437. virtual IWUResult * updateVariableByName(const char * name)
  1438. { return c->updateVariableByName(name); }
  1439. virtual void addFile(const char *fileName, StringArray *clusters, unsigned usageCount, WUFileKind fileKind, const char *graphOwner)
  1440. { c->addFile(fileName, clusters, usageCount, fileKind, graphOwner); }
  1441. virtual void noteFileRead(IDistributedFile *file)
  1442. { c->noteFileRead(file); }
  1443. virtual void releaseFile(const char *fileName)
  1444. { c->releaseFile(fileName); }
  1445. virtual void clearGraphProgress()
  1446. { c->clearGraphProgress(); }
  1447. virtual void resetBeforeGeneration()
  1448. { c->resetBeforeGeneration(); }
  1449. virtual void deleteTempFiles(const char *graph, bool deleteOwned, bool deleteJobOwned)
  1450. { c->deleteTempFiles(graph, deleteOwned, deleteJobOwned); }
  1451. virtual void deleteTemporaries()
  1452. { c->deleteTemporaries(); }
  1453. virtual void addDiskUsageStats(__int64 avgNodeUsage, unsigned minNode, __int64 minNodeUsage, unsigned maxNode, __int64 maxNodeUsage, __int64 graphId)
  1454. { c->addDiskUsageStats(avgNodeUsage, minNode, minNodeUsage, maxNode, maxNodeUsage, graphId); }
  1455. virtual IPropertyTree * getDiskUsageStats()
  1456. { return c->getDiskUsageStats(); }
  1457. virtual IPropertyTreeIterator & getFileIterator() const
  1458. { return c->getFileIterator(); }
  1459. virtual IPropertyTreeIterator & getFilesReadIterator() const
  1460. { return c->getFilesReadIterator(); }
  1461. virtual void setSnapshot(const char * value)
  1462. { c->setSnapshot(value); }
  1463. virtual void setWarningSeverity(unsigned code, ErrorSeverity severity)
  1464. { c->setWarningSeverity(code, severity); }
  1465. virtual void setTimeScheduled(const IJlibDateTime &val)
  1466. { c->setTimeScheduled(val); }
  1467. virtual void setDebugAgentListenerPort(unsigned port)
  1468. { c->setDebugAgentListenerPort(port); }
  1469. virtual void setDebugAgentListenerIP(const char * ip)
  1470. { c->setDebugAgentListenerIP(ip); }
  1471. virtual void setXmlParams(const char *params)
  1472. { c->setXmlParams(params); }
  1473. virtual void setXmlParams(IPropertyTree *tree)
  1474. { c->setXmlParams(tree); }
  1475. virtual void setHash(unsigned __int64 hash)
  1476. { c->setHash(hash); }
  1477. // ILocalWorkUnit - used for debugging etc
  1478. virtual void serialize(MemoryBuffer &tgt)
  1479. { c->serialize(tgt); }
  1480. virtual void deserialize(MemoryBuffer &src)
  1481. { c->deserialize(src); }
  1482. virtual bool switchThorQueue(const char *cluster, IQueueSwitcher *qs)
  1483. { return c->switchThorQueue(cluster,qs); }
  1484. virtual void setAllowedClusters(const char *value)
  1485. { c->setAllowedClusters(value); }
  1486. virtual IStringVal& getAllowedClusters(IStringVal &str) const
  1487. { return c->getAllowedClusters(str); }
  1488. virtual void remoteCheckAccess(IUserDescriptor *user, bool writeaccess) const
  1489. { c->remoteCheckAccess(user,writeaccess); }
  1490. virtual void setAllowAutoQueueSwitch(bool val)
  1491. { c->setAllowAutoQueueSwitch(val); }
  1492. virtual bool getAllowAutoQueueSwitch() const
  1493. { return c->getAllowAutoQueueSwitch(); }
  1494. virtual void setLibraryInformation(const char * name, unsigned interfaceHash, unsigned definitionHash)
  1495. { c->setLibraryInformation(name, interfaceHash, definitionHash); }
  1496. virtual void setResultInt(const char * name, unsigned sequence, __int64 val)
  1497. { c->setResultInt(name, sequence, val); }
  1498. virtual void setResultUInt(const char * name, unsigned sequence, unsigned __int64 val)
  1499. { c->setResultUInt(name, sequence, val); }
  1500. virtual void setResultReal(const char *name, unsigned sequence, double val)
  1501. { c->setResultReal(name, sequence, val); }
  1502. virtual void setResultVarString(const char * stepname, unsigned sequence, const char *val)
  1503. { c->setResultVarString(stepname, sequence, val); }
  1504. virtual void setResultVarUnicode(const char * stepname, unsigned sequence, UChar const *val)
  1505. { c->setResultVarUnicode(stepname, sequence, val); }
  1506. virtual void setResultString(const char * stepname, unsigned sequence, int len, const char *val)
  1507. { c->setResultString(stepname, sequence, len, val); }
  1508. virtual void setResultData(const char * stepname, unsigned sequence, int len, const void *val)
  1509. { c->setResultData(stepname, sequence, len, val); }
  1510. virtual void setResultRaw(const char * name, unsigned sequence, int len, const void *val)
  1511. { c->setResultRaw(name, sequence, len, val); }
  1512. virtual void setResultSet(const char * name, unsigned sequence, bool isAll, size32_t len, const void *val, ISetToXmlTransformer *xform)
  1513. { c->setResultSet(name, sequence, isAll, len, val, xform); }
  1514. virtual void setResultUnicode(const char * name, unsigned sequence, int len, UChar const * val)
  1515. { c->setResultUnicode(name, sequence, len, val); }
  1516. virtual void setResultBool(const char *name, unsigned sequence, bool val)
  1517. { c->setResultBool(name, sequence, val); }
  1518. virtual void setResultDecimal(const char *name, unsigned sequence, int len, int precision, bool isSigned, const void *val)
  1519. { c->setResultDecimal(name, sequence, len, precision, isSigned, val); }
  1520. virtual void setResultDataset(const char * name, unsigned sequence, size32_t len, const void *val, unsigned numRows, bool extend)
  1521. { c->setResultDataset(name, sequence, len, val, numRows, extend); }
  1522. };
  1523. class CLocalWUAssociated : public CInterface, implements IConstWUAssociatedFile
  1524. {
  1525. Owned<IPropertyTree> p;
  1526. public:
  1527. IMPLEMENT_IINTERFACE;
  1528. CLocalWUAssociated(IPropertyTree *p);
  1529. virtual WUFileType getType() const;
  1530. virtual IStringVal & getDescription(IStringVal & ret) const;
  1531. virtual IStringVal & getIp(IStringVal & ret) const;
  1532. virtual IStringVal & getName(IStringVal & ret) const;
  1533. virtual IStringVal & getNameTail(IStringVal & ret) const;
  1534. virtual unsigned getCrc() const;
  1535. };
  1536. class CLocalWUQuery : public CInterface, implements IWUQuery
  1537. {
  1538. Owned<IPropertyTree> p;
  1539. mutable IArrayOf<IConstWUAssociatedFile> associated;
  1540. mutable CriticalSection crit;
  1541. mutable bool associatedCached;
  1542. private:
  1543. void addSpecialCaseAssociated(WUFileType type, const char * propname, unsigned crc) const;
  1544. void loadAssociated() const;
  1545. public:
  1546. IMPLEMENT_IINTERFACE;
  1547. CLocalWUQuery(IPropertyTree *p);
  1548. virtual WUQueryType getQueryType() const;
  1549. virtual IStringVal& getQueryText(IStringVal &str) const;
  1550. virtual IStringVal& getQueryShortText(IStringVal &str) const;
  1551. virtual IStringVal& getQueryName(IStringVal &str) const;
  1552. virtual IStringVal & getQueryMainDefinition(IStringVal & str) const;
  1553. virtual IStringVal& getQueryDllName(IStringVal &str) const;
  1554. virtual unsigned getQueryDllCrc() const;
  1555. virtual IStringVal& getQueryCppName(IStringVal &str) const;
  1556. virtual IStringVal& getQueryResTxtName(IStringVal &str) const;
  1557. virtual IConstWUAssociatedFile * getAssociatedFile(WUFileType type, unsigned index) const;
  1558. virtual IConstWUAssociatedFileIterator& getAssociatedFiles() const;
  1559. virtual void setQueryType(WUQueryType qt);
  1560. virtual void setQueryText(const char *pstr);
  1561. virtual void setQueryName(const char *);
  1562. virtual void setQueryMainDefinition(const char * str);
  1563. virtual void addAssociatedFile(WUFileType type, const char * name, const char * ip, const char * desc, unsigned crc);
  1564. virtual void removeAssociatedFiles();
  1565. virtual void removeAssociatedFile(WUFileType type, const char * name, const char * desc);
  1566. };
  1567. class CLocalWUWebServicesInfo : public CInterface, implements IWUWebServicesInfo
  1568. {
  1569. Owned<IPropertyTree> p;
  1570. mutable CriticalSection crit;
  1571. private:
  1572. public:
  1573. IMPLEMENT_IINTERFACE;
  1574. CLocalWUWebServicesInfo(IPropertyTree *p);
  1575. virtual IStringVal& getModuleName(IStringVal &str) const;
  1576. virtual IStringVal& getAttributeName(IStringVal &str) const;
  1577. virtual IStringVal& getDefaultName(IStringVal &str) const;
  1578. virtual IStringVal& getInfo(const char *name, IStringVal &str) const;
  1579. virtual IStringVal& getText(const char *name, IStringVal &str) const;
  1580. virtual unsigned getWebServicesCRC() const;
  1581. virtual void setModuleName(const char *);
  1582. virtual void setAttributeName(const char *);
  1583. virtual void setDefaultName(const char *);
  1584. virtual void setInfo(const char *name, const char *info);
  1585. virtual void setText(const char *name, const char *info);
  1586. virtual void setWebServicesCRC(unsigned);
  1587. };
  1588. class CLocalWURoxieQueryInfo : public CInterface, implements IWURoxieQueryInfo
  1589. {
  1590. Owned<IPropertyTree> p;
  1591. mutable CriticalSection crit;
  1592. private:
  1593. public:
  1594. IMPLEMENT_IINTERFACE;
  1595. CLocalWURoxieQueryInfo(IPropertyTree *p);
  1596. virtual IStringVal& getQueryInfo(IStringVal &str) const;
  1597. virtual IStringVal& getDefaultPackageInfo(IStringVal &str) const;
  1598. virtual IStringVal& getRoxieClusterName(IStringVal &str) const;
  1599. virtual IStringVal& getWuid(IStringVal &str) const;
  1600. virtual void setQueryInfo(const char *info);
  1601. virtual void setDefaultPackageInfo(const char *, int len);
  1602. virtual void setRoxieClusterName(const char *str);
  1603. virtual void setWuid(const char *str);
  1604. };
  1605. class CLocalWUResult : public CInterface, implements IWUResult
  1606. {
  1607. friend class CLocalWorkUnit;
  1608. mutable CriticalSection crit;
  1609. Owned<IPropertyTree> p;
  1610. Owned<IProperties> xmlns;
  1611. void getSchema(TypeInfoArray &types, StringAttrArray &names, IStringVal * ecl=NULL) const;
  1612. public:
  1613. IMPLEMENT_IINTERFACE;
  1614. CLocalWUResult(IPropertyTree *props);
  1615. ~CLocalWUResult() { try { p.clear(); } catch (IException *E) {E->Release();}}
  1616. virtual WUResultStatus getResultStatus() const;
  1617. virtual IStringVal& getResultName(IStringVal &str) const;
  1618. virtual int getResultSequence() const;
  1619. virtual bool isResultScalar() const;
  1620. virtual IStringVal& getResultXml(IStringVal &str, bool hidePasswords) const;
  1621. virtual unsigned getResultFetchSize() const;
  1622. virtual __int64 getResultTotalRowCount() const;
  1623. virtual __int64 getResultRowCount() const;
  1624. virtual void getResultDataset(IStringVal & ecl, IStringVal & defs) const;
  1625. virtual IStringVal& getResultLogicalName(IStringVal &ecl) const;
  1626. virtual IStringVal& getResultKeyField(IStringVal& ecl) const;
  1627. virtual unsigned getResultRequestedRows() const;
  1628. virtual __int64 getResultInt() const;
  1629. virtual bool getResultBool() const;
  1630. virtual double getResultReal() const;
  1631. virtual IStringVal& getResultString(IStringVal & str, bool hidePassword) const;
  1632. virtual IDataVal& getResultRaw(IDataVal & data, IXmlToRawTransformer * xmlTransformer, ICsvToRawTransformer * csvTransformer) const;
  1633. virtual IDataVal& getResultUnicode(IDataVal & data) const;
  1634. virtual void getResultDecimal(void * val, unsigned length, unsigned precision, bool isSigned) const;
  1635. virtual IStringVal& getResultEclSchema(IStringVal & str) const;
  1636. virtual __int64 getResultRawSize(IXmlToRawTransformer * xmlTransformer, ICsvToRawTransformer * csvTransformer) const;
  1637. virtual IDataVal& getResultRaw(IDataVal & data, __int64 from, __int64 length, IXmlToRawTransformer * xmlTransformer, ICsvToRawTransformer * csvTransformer) const;
  1638. virtual IStringVal& getResultRecordSizeEntry(IStringVal & str) const;
  1639. virtual IStringVal& getResultTransformerEntry(IStringVal & str) const;
  1640. virtual __int64 getResultRowLimit() const;
  1641. virtual IStringVal& getResultFilename(IStringVal & str) const;
  1642. virtual WUResultFormat getResultFormat() const;
  1643. virtual unsigned getResultHash() const;
  1644. virtual bool getResultIsAll() const;
  1645. virtual IProperties *queryResultXmlns();
  1646. virtual IStringVal& getResultFieldOpt(const char *name, IStringVal &str) const;
  1647. // interface IWUResult
  1648. virtual void setResultStatus(WUResultStatus status);
  1649. virtual void setResultName(const char *name);
  1650. virtual void setResultSequence(unsigned seq);
  1651. virtual void setResultSchemaRaw(unsigned len, const void *schema);
  1652. virtual void setResultScalar(bool isScalar);
  1653. virtual void setResultRaw(unsigned len, const void *xml, WUResultFormat format);
  1654. virtual void setResultFetchSize(unsigned rows); // 0 means file-loaded
  1655. virtual void setResultTotalRowCount(__int64 rows); // -1 means unknown
  1656. virtual void setResultRowCount(__int64 rows);
  1657. virtual void setResultDataset(const char *ecl, const char *defs);
  1658. virtual void setResultLogicalName(const char *logicalName);
  1659. virtual void setResultKeyField(const char * name);
  1660. virtual void setResultRequestedRows(unsigned req);
  1661. virtual void setResultRecordSizeEntry(const char * val);
  1662. virtual void setResultTransformerEntry(const char * val);
  1663. virtual void setResultInt(__int64 val);
  1664. virtual void setResultReal(double val);
  1665. virtual void setResultBool(bool val);
  1666. virtual void setResultString(const char * val, unsigned length);
  1667. virtual void setResultUnicode(const void * val, unsigned length);
  1668. virtual void setResultData(const void * val, unsigned length);
  1669. virtual void setResultDecimal(const void * val, unsigned length);
  1670. virtual void addResultRaw(unsigned len, const void * data, WUResultFormat format);
  1671. virtual void setResultRowLimit(__int64 value);
  1672. virtual void setResultFilename(const char * name);
  1673. virtual void setResultUInt(unsigned __int64 val);
  1674. virtual void setResultIsAll(bool value);
  1675. virtual void setResultFormat(WUResultFormat format);
  1676. virtual void setResultXML(const char *val);
  1677. virtual void setResultRow(unsigned len, const void * data);
  1678. virtual void setResultXmlns(const char *prefix, const char *uri);
  1679. virtual void setResultFieldOpt(const char *name, const char *value);
  1680. virtual IPropertyTree *queryPTree() { return p; }
  1681. };
  1682. class CLocalWUPlugin : public CInterface, implements IWUPlugin
  1683. {
  1684. Owned<IPropertyTree> p;
  1685. public:
  1686. IMPLEMENT_IINTERFACE;
  1687. CLocalWUPlugin(IPropertyTree *p);
  1688. virtual IStringVal& getPluginName(IStringVal &str) const;
  1689. virtual IStringVal& getPluginVersion(IStringVal &str) const;
  1690. virtual void setPluginName(const char *str);
  1691. virtual void setPluginVersion(const char *str);
  1692. };
  1693. class CLocalWULibrary : public CInterface, implements IWULibrary
  1694. {
  1695. Owned<IPropertyTree> p;
  1696. public:
  1697. IMPLEMENT_IINTERFACE;
  1698. CLocalWULibrary(IPropertyTree *p);
  1699. virtual IStringVal & getName(IStringVal & str) const;
  1700. virtual void setName(const char * str);
  1701. };
  1702. class CLocalWUGraph : public CInterface, implements IWUGraph
  1703. {
  1704. const CLocalWorkUnit &owner;
  1705. Owned<IPropertyTree> p;
  1706. mutable Owned<IPropertyTree> graph; // cached copy of graph xgmml
  1707. mutable Linked<IConstWUGraphProgress> progress;
  1708. unsigned wuidVersion;
  1709. void mergeProgress(IPropertyTree &tree, IPropertyTree &progressTree, const unsigned &progressV) const;
  1710. public:
  1711. IMPLEMENT_IINTERFACE;
  1712. CLocalWUGraph(const CLocalWorkUnit &owner, IPropertyTree *p);
  1713. virtual IStringVal & getXGMML(IStringVal & ret, bool mergeProgress) const;
  1714. virtual IStringVal & getName(IStringVal & ret) const;
  1715. virtual IStringVal & getLabel(IStringVal & ret) const;
  1716. virtual IStringVal & getTypeName(IStringVal & ret) const;
  1717. virtual WUGraphType getType() const;
  1718. virtual WUGraphState getState() const;
  1719. virtual IPropertyTree * getXGMMLTree(bool mergeProgress) const;
  1720. virtual IPropertyTree * getXGMMLTreeRaw() const;
  1721. virtual void setName(const char *str);
  1722. virtual void setLabel(const char *str);
  1723. virtual void setType(WUGraphType type);
  1724. virtual void setXGMML(const char *str);
  1725. virtual void setXGMMLTree(IPropertyTree * tree, bool compress=true);
  1726. };
  1727. class CLocalWUException : public CInterface, implements IWUException
  1728. {
  1729. Owned<IPropertyTree> p;
  1730. public:
  1731. IMPLEMENT_IINTERFACE;
  1732. CLocalWUException(IPropertyTree *p);
  1733. virtual IStringVal& getExceptionSource(IStringVal &str) const;
  1734. virtual IStringVal& getExceptionMessage(IStringVal &str) const;
  1735. virtual unsigned getExceptionCode() const;
  1736. virtual ErrorSeverity getSeverity() const;
  1737. virtual IStringVal & getTimeStamp(IStringVal & dt) const;
  1738. virtual IStringVal & getExceptionFileName(IStringVal & str) const;
  1739. virtual unsigned getExceptionLineNo() const;
  1740. virtual unsigned getExceptionColumn() const;
  1741. virtual unsigned getSequence() const;
  1742. virtual void setExceptionSource(const char *str);
  1743. virtual void setExceptionMessage(const char *str);
  1744. virtual void setExceptionCode(unsigned code);
  1745. virtual void setSeverity(ErrorSeverity level);
  1746. virtual void setTimeStamp(const char * dt);
  1747. virtual void setExceptionFileName(const char *str);
  1748. virtual void setExceptionLineNo(unsigned r);
  1749. virtual void setExceptionColumn(unsigned c);
  1750. };
  1751. //==========================================================================================
  1752. extern WORKUNIT_API bool isSpecialResultSequence(unsigned sequence)
  1753. {
  1754. switch (sequence)
  1755. {
  1756. case ResultSequenceInternal:
  1757. case ResultSequenceOnce:
  1758. case ResultSequencePersist:
  1759. case ResultSequenceStored:
  1760. return true;
  1761. default:
  1762. assertex(sequence <= INT_MAX);
  1763. return false;
  1764. }
  1765. }
  1766. class CConstWUArrayIterator : public CInterface, implements IConstWorkUnitIterator
  1767. {
  1768. IArrayOf<IPropertyTree> trees;
  1769. Owned<IConstWorkUnitInfo> cur;
  1770. unsigned curTreeNum;
  1771. void setCurrent()
  1772. {
  1773. cur.setown(new CLightweightWorkunitInfo(trees.item(curTreeNum)));
  1774. }
  1775. public:
  1776. IMPLEMENT_IINTERFACE;
  1777. CConstWUArrayIterator(IArrayOf<IPropertyTree> &_trees)
  1778. {
  1779. ForEachItemIn(t, _trees)
  1780. trees.append(*LINK(&_trees.item(t)));
  1781. curTreeNum = 0;
  1782. }
  1783. bool first()
  1784. {
  1785. curTreeNum = 0;
  1786. return next();
  1787. }
  1788. bool isValid()
  1789. {
  1790. return (NULL != cur.get());
  1791. }
  1792. bool next()
  1793. {
  1794. if (curTreeNum >= trees.ordinality())
  1795. {
  1796. cur.clear();
  1797. return false;
  1798. }
  1799. setCurrent();
  1800. ++curTreeNum;
  1801. return true;
  1802. }
  1803. IConstWorkUnitInfo & query() { return *cur; }
  1804. };
  1805. //==========================================================================================
  1806. class CStringArrayIterator : public CInterface, implements IStringIterator
  1807. {
  1808. StringArray strings;
  1809. unsigned idx;
  1810. public:
  1811. IMPLEMENT_IINTERFACE;
  1812. CStringArrayIterator() { idx = 0; };
  1813. void append(const char *str) { strings.append(str); }
  1814. virtual bool first() { idx = 0; return strings.isItem(idx); }
  1815. virtual bool next() { idx ++; return strings.isItem(idx); }
  1816. virtual bool isValid() { return strings.isItem(idx); }
  1817. virtual IStringVal & str(IStringVal &s) { s.set(strings.item(idx)); return s; }
  1818. };
  1819. class CCachedJobNameIterator : public CInterface, implements IStringIterator
  1820. {
  1821. Owned<IPropertyTreeIterator> it;
  1822. public:
  1823. IMPLEMENT_IINTERFACE;
  1824. CCachedJobNameIterator(IPropertyTreeIterator *p) : it(p) {};
  1825. virtual bool first() { return it->first(); }
  1826. virtual bool next() { return it->next(); }
  1827. virtual bool isValid() { return it->isValid(); }
  1828. virtual IStringVal & str(IStringVal &s) { s.set(it->query().queryName()+1); return s; }
  1829. };
  1830. class CEmptyStringIterator : public CInterface, implements IStringIterator
  1831. {
  1832. public:
  1833. IMPLEMENT_IINTERFACE;
  1834. virtual bool first() { return false; }
  1835. virtual bool next() { return false; }
  1836. virtual bool isValid() { return false; }
  1837. virtual IStringVal & str(IStringVal &s) { s.clear(); return s; }
  1838. };
  1839. mapEnums workunitSortFields[] =
  1840. {
  1841. { WUSFuser, "@submitID" },
  1842. { WUSFcluster, "@clusterName" },
  1843. { WUSFjob, "@jobName" },
  1844. { WUSFstate, "@state" },
  1845. { WUSFpriority, "@priorityClass" },
  1846. { WUSFprotected, "@protected" },
  1847. { WUSFwuid, "@" },
  1848. { WUSFecl, "Query/Text" },
  1849. { WUSFfileread, "FilesRead/File/@name" },
  1850. { WUSFroxiecluster, "RoxieQueryInfo/@roxieClusterName" },
  1851. { WUSFtotalthortime, "Statistics/Statistic[@c='summary'][@creator='thor'][@kind='TimeElapsed']/@value|"
  1852. "Statistics/Statistic[@c='summary'][@creator='hthor'][@kind='TimeElapsed']/@value|"
  1853. "Statistics/Statistic[@c='summary'][@creator='roxie'][@kind='TimeElapsed']/@value|"
  1854. "Statistics/Statistic[@desc='Total thor time']/@value|"
  1855. "Timings/Timing[@name='Total thor time']/@duration" //Use Statistics first. If not found, use Timings
  1856. },
  1857. { WUSFterm, NULL }
  1858. };
  1859. mapEnums querySortFields[] =
  1860. {
  1861. { WUQSFId, "@id" },
  1862. { WUQSFwuid, "@wuid" },
  1863. { WUQSFname, "@name" },
  1864. { WUQSFdll, "@dll" },
  1865. { WUQSFmemoryLimit, "@memoryLimit" },
  1866. { WUQSFmemoryLimitHi, "@memoryLimit" },
  1867. { WUQSFtimeLimit, "@timeLimit" },
  1868. { WUQSFtimeLimitHi, "@timeLimit" },
  1869. { WUQSFwarnTimeLimit, "@warnTimeLimit" },
  1870. { WUQSFwarnTimeLimitHi, "@warnTimeLimit" },
  1871. { WUQSFpriority, "@priority" },
  1872. { WUQSFpriorityHi, "@priority" },
  1873. { WUQSFQuerySet, "@querySetId" },
  1874. { WUQSFActivited, "@activated" },
  1875. { WUQSFSuspendedByUser, "@suspended" },
  1876. { WUQSFLibrary, "Library"},
  1877. { WUQSFterm, NULL }
  1878. };
  1879. class asyncRemoveDllWorkItem: public CInterface, implements IWorkQueueItem // class only used in asyncRemoveDll
  1880. {
  1881. StringAttr name;
  1882. public:
  1883. IMPLEMENT_IINTERFACE;
  1884. asyncRemoveDllWorkItem(const char * _name) : name(_name)
  1885. {
  1886. }
  1887. void execute()
  1888. {
  1889. PROGLOG("WU removeDll %s", name.get());
  1890. queryDllServer().removeDll(name, true, true); // <name>, removeDlls=true, removeDirectory=true
  1891. }
  1892. };
  1893. class asyncRemoveRemoteFileWorkItem: public CInterface, implements IWorkQueueItem // class only used in asyncRemoveFile
  1894. {
  1895. RemoteFilename name;
  1896. public:
  1897. IMPLEMENT_IINTERFACE;
  1898. asyncRemoveRemoteFileWorkItem(const char * _ip, const char * _name)
  1899. {
  1900. SocketEndpoint ep(_ip);
  1901. name.setPath(ep, _name);
  1902. }
  1903. void execute()
  1904. {
  1905. Owned<IFile> file = createIFile(name);
  1906. PROGLOG("WU removeDll %s",file->queryFilename());
  1907. file->remove();
  1908. }
  1909. };
  1910. //==========================================================================================
  1911. class CConstQuerySetQueryIterator : public CInterface, implements IConstQuerySetQueryIterator
  1912. {
  1913. IArrayOf<IPropertyTree> trees;
  1914. unsigned index;
  1915. public:
  1916. IMPLEMENT_IINTERFACE;
  1917. CConstQuerySetQueryIterator(IArrayOf<IPropertyTree> &_trees)
  1918. {
  1919. ForEachItemIn(t, _trees)
  1920. trees.append(*LINK(&_trees.item(t)));
  1921. index = 0;
  1922. }
  1923. ~CConstQuerySetQueryIterator()
  1924. {
  1925. trees.kill();
  1926. }
  1927. bool first()
  1928. {
  1929. index = 0;
  1930. return (trees.ordinality()!=0);
  1931. }
  1932. bool next()
  1933. {
  1934. index++;
  1935. return (index<trees.ordinality());
  1936. }
  1937. bool isValid()
  1938. {
  1939. return (index<trees.ordinality());
  1940. }
  1941. IPropertyTree &query()
  1942. {
  1943. return trees.item(index);
  1944. }
  1945. };
  1946. class CSecurityCache
  1947. {
  1948. };
  1949. class CConstWUIterator : public CInterface, implements IConstWorkUnitIterator
  1950. {
  1951. public:
  1952. IMPLEMENT_IINTERFACE;
  1953. CConstWUIterator(IPropertyTreeIterator *_ptreeIter)
  1954. : ptreeIter(_ptreeIter)
  1955. {
  1956. }
  1957. bool first()
  1958. {
  1959. if (!ptreeIter->first())
  1960. {
  1961. cur.clear();
  1962. return false;
  1963. }
  1964. cur.setown(new CLightweightWorkunitInfo(ptreeIter->query()));
  1965. return true;
  1966. }
  1967. bool isValid()
  1968. {
  1969. return (NULL != cur.get());
  1970. }
  1971. bool next()
  1972. {
  1973. if (!ptreeIter->next())
  1974. {
  1975. cur.clear();
  1976. return false;
  1977. }
  1978. cur.setown(new CLightweightWorkunitInfo(ptreeIter->query()));
  1979. return true;
  1980. }
  1981. IConstWorkUnitInfo & query() { return *cur; }
  1982. private:
  1983. Owned<IConstWorkUnitInfo> cur;
  1984. Owned<IPropertyTreeIterator> ptreeIter;
  1985. };
  1986. class CSecureConstWUIterator : public CInterfaceOf<IConstWorkUnitIterator>
  1987. {
  1988. public:
  1989. CSecureConstWUIterator(IConstWorkUnitIterator *_parent, ISecManager *_secmgr=NULL, ISecUser *_secuser=NULL)
  1990. : parent(_parent), secmgr(_secmgr), secuser(_secuser)
  1991. {
  1992. assertex(_secuser && _secmgr);
  1993. }
  1994. bool first()
  1995. {
  1996. if (!parent->first())
  1997. return false;
  1998. return getNext();
  1999. }
  2000. bool next()
  2001. {
  2002. if (!parent->next())
  2003. return false;
  2004. return getNext();
  2005. }
  2006. virtual bool isValid()
  2007. {
  2008. return parent->isValid();
  2009. }
  2010. virtual IConstWorkUnitInfo &query()
  2011. {
  2012. return parent->query();
  2013. }
  2014. private:
  2015. Owned<IConstWorkUnitIterator> parent;
  2016. MapStringTo<int> scopePermissions;
  2017. Linked<ISecManager> secmgr;
  2018. Linked<ISecUser> secuser;
  2019. bool getNext() // scan for a workunit with permissions
  2020. {
  2021. do
  2022. {
  2023. const char *scopeName = parent->query().queryWuScope();
  2024. if (!scopeName || !*scopeName || checkScope(scopeName))
  2025. return true;
  2026. } while (parent->next());
  2027. return false;
  2028. }
  2029. bool checkScope(const char *scopeName)
  2030. {
  2031. int *perms = scopePermissions.getValue(scopeName);
  2032. int perm;
  2033. if (!perms)
  2034. {
  2035. perm = secmgr->authorizeWorkunitScope(*secuser, scopeName);
  2036. scopePermissions.setValue(scopeName, perm);
  2037. }
  2038. else
  2039. perm = *perms;
  2040. return perm >= SecAccess_Read;
  2041. }
  2042. };
  2043. #define WUID_VERSION 2 // recorded in each wuid created, useful for bkwd compat. checks
  2044. CWorkUnitFactory::CWorkUnitFactory()
  2045. {
  2046. }
  2047. CWorkUnitFactory::~CWorkUnitFactory()
  2048. {
  2049. }
  2050. IWorkUnit* CWorkUnitFactory::createNamedWorkUnit(const char *wuid, const char *app, const char *scope, ISecManager *secmgr, ISecUser *secuser)
  2051. {
  2052. checkWuScopeSecAccess(scope, secmgr, secuser, SecAccess_Write, "Create", true, true);
  2053. Owned<CLocalWorkUnit> cw = _createWorkUnit(wuid, secmgr, secuser);
  2054. if (scope)
  2055. cw->setWuScope(scope); // Note - this may check access rights and throw exception. Is that correct? We might prefer to only check access once, and this will check on the lock too...
  2056. IWorkUnit* ret = &cw->lockRemote(false); // Note - this may throw exception if user does not have rights.
  2057. ret->setDebugValue("CREATED_BY", app, true);
  2058. ret->setDebugValue("CREATED_FOR", scope, true);
  2059. return ret;
  2060. }
  2061. IWorkUnit* CWorkUnitFactory::createWorkUnit(const char *app, const char *scope, ISecManager *secmgr, ISecUser *secuser)
  2062. {
  2063. StringBuffer wuid("W");
  2064. char result[32];
  2065. time_t ltime;
  2066. time( &ltime );
  2067. tm *today = localtime( &ltime ); // MORE - this is not threadsafe. But I probably don't care that much!
  2068. strftime(result, sizeof(result), "%Y%m%d-%H%M%S", today);
  2069. wuid.append(result);
  2070. if (workUnitTraceLevel > 1)
  2071. PrintLog("createWorkUnit created %s", wuid.str());
  2072. IWorkUnit* ret = createNamedWorkUnit(wuid.str(), app, scope, secmgr, secuser);
  2073. if (workUnitTraceLevel > 1)
  2074. PrintLog("createWorkUnit created %s", ret->queryWuid());
  2075. addTimeStamp(ret, SSTglobal, NULL, StWhenCreated);
  2076. return ret;
  2077. }
  2078. bool CWorkUnitFactory::deleteWorkUnit(const char * wuid, ISecManager *secmgr, ISecUser *secuser)
  2079. {
  2080. if (workUnitTraceLevel > 1)
  2081. PrintLog("deleteWorkUnit %s", wuid);
  2082. StringBuffer wuRoot;
  2083. getXPath(wuRoot, wuid);
  2084. Owned<CLocalWorkUnit> cw = _updateWorkUnit(wuid, secmgr, secuser);
  2085. if (!checkWuSecAccess(*cw.get(), secmgr, secuser, SecAccess_Full, "delete", true, true))
  2086. return false;
  2087. try
  2088. {
  2089. cw->cleanupAndDelete(true, true);
  2090. }
  2091. catch (IException *E)
  2092. {
  2093. StringBuffer s;
  2094. LOG(MCexception(E, MSGCLS_warning), E, s.append("Exception during deleteWorkUnit: ").append(wuid).str());
  2095. E->Release();
  2096. return false;
  2097. }
  2098. removeWorkUnitFromAllQueues(wuid); //known active workunits wouldn't make it this far
  2099. return true;
  2100. }
  2101. IConstWorkUnit* CWorkUnitFactory::openWorkUnit(const char *wuid, bool lock, ISecManager *secmgr, ISecUser *secuser)
  2102. {
  2103. StringBuffer wuidStr(wuid);
  2104. wuidStr.trim();
  2105. if (wuidStr.length() && ('w' == wuidStr.charAt(0)))
  2106. wuidStr.setCharAt(0, 'W');
  2107. if (!wuidStr.length() || ('W' != wuidStr.charAt(0)))
  2108. {
  2109. if (workUnitTraceLevel > 1)
  2110. PrintLog("openWorkUnit %s invalid WUID", nullText(wuidStr.str()));
  2111. return NULL;
  2112. }
  2113. if (workUnitTraceLevel > 1)
  2114. PrintLog("openWorkUnit %s", wuidStr.str());
  2115. Owned<IConstWorkUnit> wu = _openWorkUnit(wuid, lock, secmgr, secuser);
  2116. if (wu)
  2117. {
  2118. if (!checkWuSecAccess(*wu, secmgr, secuser, SecAccess_Read, "opening", true, true))
  2119. return NULL; // Actually throws exception on failure, so won't reach here
  2120. return wu.getClear();
  2121. }
  2122. else
  2123. {
  2124. if (workUnitTraceLevel > 0)
  2125. PrintLog("openWorkUnit %s not found", wuidStr.str());
  2126. return NULL;
  2127. }
  2128. }
  2129. IWorkUnit* CWorkUnitFactory::updateWorkUnit(const char *wuid, ISecManager *secmgr, ISecUser *secuser)
  2130. {
  2131. if (workUnitTraceLevel > 1)
  2132. PrintLog("updateWorkUnit %s", wuid);
  2133. Owned<CLocalWorkUnit> wu = _updateWorkUnit(wuid, secmgr, secuser);
  2134. if (wu)
  2135. {
  2136. if (!checkWuSecAccess(*wu.get(), secmgr, secuser, SecAccess_Write, "updating", true, true))
  2137. return NULL;
  2138. return &wu->lockRemote(false);
  2139. }
  2140. else
  2141. {
  2142. if (workUnitTraceLevel > 0)
  2143. PrintLog("updateWorkUnit %s not found", wuid);
  2144. return NULL;
  2145. }
  2146. }
  2147. int CWorkUnitFactory::setTracingLevel(int newLevel)
  2148. {
  2149. if (newLevel)
  2150. PrintLog("Setting workunit trace level to %d", newLevel);
  2151. int level = workUnitTraceLevel;
  2152. workUnitTraceLevel = newLevel;
  2153. return level;
  2154. }
  2155. void CWorkUnitFactory::descheduleAllWorkUnits(ISecManager *secmgr, ISecUser *secuser)
  2156. {
  2157. Owned<IRemoteConnection> conn = querySDS().connect("/Schedule", myProcessSession(), RTM_LOCK_WRITE, SDS_LOCK_TIMEOUT);
  2158. if(!conn) return;
  2159. Owned<IPropertyTree> root(conn->queryRoot()->getBranch("."));
  2160. KeptAtomTable entries;
  2161. Owned<IPropertyTreeIterator> iter(root->getElements("*/*/*/*"));
  2162. StringBuffer wuid;
  2163. for(iter->first(); iter->isValid(); iter->next())
  2164. {
  2165. char const * entry = iter->query().queryName();
  2166. if(!entries.find(entry))
  2167. {
  2168. entries.addAtom(entry);
  2169. ncnameUnescape(entry, wuid.clear());
  2170. Owned<IWorkUnit> wu = updateWorkUnit(wuid, secmgr, secuser);
  2171. if(wu && (wu->getState() == WUStateWait))
  2172. wu->setState(WUStateCompleted);
  2173. }
  2174. }
  2175. bool more;
  2176. do more = root->removeProp("*"); while(more);
  2177. }
  2178. IConstQuerySetQueryIterator* CWorkUnitFactory::getQuerySetQueriesSorted( WUQuerySortField *sortorder, // list of fields to sort by (terminated by WUSFterm)
  2179. WUQuerySortField *filters, // NULL or list of fields to filter on (terminated by WUSFterm)
  2180. const void *filterbuf, // (appended) string values for filters
  2181. unsigned startoffset,
  2182. unsigned maxnum,
  2183. __int64 *cachehint,
  2184. unsigned *total,
  2185. const MapStringTo<bool> *_subset)
  2186. {
  2187. struct PostFilters
  2188. {
  2189. WUQueryFilterBoolean activatedFilter;
  2190. WUQueryFilterBoolean suspendedByUserFilter;
  2191. PostFilters()
  2192. {
  2193. activatedFilter = WUQFSAll;
  2194. suspendedByUserFilter = WUQFSAll;
  2195. };
  2196. } postFilters;
  2197. class CQuerySetQueriesPager : public CSimpleInterface, implements IElementsPager
  2198. {
  2199. StringAttr querySet;
  2200. StringAttr xPath;
  2201. StringAttr sortOrder;
  2202. PostFilters postFilters;
  2203. StringArray unknownAttributes;
  2204. const MapStringTo<bool> *subset;
  2205. void populateQueryTree(const IPropertyTree* querySetTree, IPropertyTree* queryTree)
  2206. {
  2207. const char* querySetId = querySetTree->queryProp("@id");
  2208. VStringBuffer path("Query%s", xPath.get());
  2209. Owned<IPropertyTreeIterator> iter = querySetTree->getElements(path.str());
  2210. ForEach(*iter)
  2211. {
  2212. IPropertyTree &query = iter->query();
  2213. bool activated = false;
  2214. const char* queryId = query.queryProp("@id");
  2215. if (queryId && *queryId)
  2216. {
  2217. if (subset)
  2218. {
  2219. VStringBuffer match("%s/%s", querySetId, queryId);
  2220. if (!subset->getValue(match))
  2221. continue;
  2222. }
  2223. VStringBuffer aliasXPath("Alias[@id='%s']", queryId);
  2224. IPropertyTree *alias = querySetTree->queryPropTree(aliasXPath.str());
  2225. if (alias)
  2226. activated = true;
  2227. }
  2228. if (activated && (postFilters.activatedFilter == WUQFSNo))
  2229. continue;
  2230. if (!activated && (postFilters.activatedFilter == WUQFSYes))
  2231. continue;
  2232. if ((postFilters.suspendedByUserFilter == WUQFSNo) && query.hasProp(getEnumText(WUQSFSuspendedByUser,querySortFields)))
  2233. continue;
  2234. if ((postFilters.suspendedByUserFilter == WUQFSYes) && !query.hasProp(getEnumText(WUQSFSuspendedByUser,querySortFields)))
  2235. continue;
  2236. IPropertyTree *queryWithSetId = queryTree->addPropTree("Query", createPTreeFromIPT(&query));
  2237. queryWithSetId->setProp("@querySetId", querySetId);
  2238. queryWithSetId->setPropBool("@activated", activated);
  2239. }
  2240. }
  2241. IRemoteConnection* populateQueryTree(IPropertyTree* queryTree)
  2242. {
  2243. StringBuffer querySetXPath("QuerySets");
  2244. if (!querySet.isEmpty())
  2245. querySetXPath.appendf("/QuerySet[@id=\"%s\"]", querySet.get());
  2246. Owned<IRemoteConnection> conn = querySDS().connect(querySetXPath.str(), myProcessSession(), 0, SDS_LOCK_TIMEOUT);
  2247. if (!conn)
  2248. return NULL;
  2249. if (querySet.isEmpty())
  2250. {
  2251. Owned<IPropertyTreeIterator> querySetIter = conn->queryRoot()->getElements("*");
  2252. ForEach(*querySetIter)
  2253. populateQueryTree(&querySetIter->query(), queryTree);
  2254. }
  2255. else
  2256. populateQueryTree(conn->queryRoot(), queryTree);
  2257. return conn.getClear();
  2258. }
  2259. public:
  2260. IMPLEMENT_IINTERFACE_USING(CSimpleInterface);
  2261. CQuerySetQueriesPager(const char* _querySet, const char* _xPath, const char *_sortOrder, PostFilters& _postFilters, StringArray& _unknownAttributes, const MapStringTo<bool> *_subset)
  2262. : querySet(_querySet), xPath(_xPath), sortOrder(_sortOrder), subset(_subset)
  2263. {
  2264. postFilters.activatedFilter = _postFilters.activatedFilter;
  2265. postFilters.suspendedByUserFilter = _postFilters.suspendedByUserFilter;
  2266. ForEachItemIn(x, _unknownAttributes)
  2267. unknownAttributes.append(_unknownAttributes.item(x));
  2268. }
  2269. virtual IRemoteConnection* getElements(IArrayOf<IPropertyTree> &elements)
  2270. {
  2271. Owned<IPropertyTree> elementTree = createPTree("Queries");
  2272. Owned<IRemoteConnection> conn = populateQueryTree(elementTree);
  2273. if (!conn)
  2274. return NULL;
  2275. Owned<IPropertyTreeIterator> iter = elementTree->getElements("*");
  2276. if (!iter)
  2277. return NULL;
  2278. sortElements(iter, sortOrder.get(), NULL, NULL, unknownAttributes, elements);
  2279. return conn.getClear();
  2280. }
  2281. };
  2282. StringAttr querySet;
  2283. StringBuffer xPath;
  2284. StringBuffer so;
  2285. StringArray unknownAttributes;
  2286. if (filters)
  2287. {
  2288. const char *fv = (const char *)filterbuf;
  2289. for (unsigned i=0;filters[i]!=WUQSFterm;i++) {
  2290. int fmt = filters[i];
  2291. int subfmt = (fmt&0xff);
  2292. if (subfmt==WUQSFQuerySet)
  2293. querySet.set(fv);
  2294. else if ((subfmt==WUQSFmemoryLimit) || (subfmt==WUQSFtimeLimit) || (subfmt==WUQSFwarnTimeLimit) || (subfmt==WUQSFpriority))
  2295. xPath.append('[').append(getEnumText(subfmt,querySortFields)).append(">=").append(fv).append("]");
  2296. else if ((subfmt==WUQSFmemoryLimitHi) || (subfmt==WUQSFtimeLimitHi) || (subfmt==WUQSFwarnTimeLimitHi) || (subfmt==WUQSFpriorityHi))
  2297. xPath.append('[').append(getEnumText(subfmt,querySortFields)).append("<=").append(fv).append("]");
  2298. else if (subfmt==WUQSFActivited)
  2299. postFilters.activatedFilter = (WUQueryFilterBoolean) atoi(fv);
  2300. else if (subfmt==WUQSFSuspendedByUser)
  2301. postFilters.suspendedByUserFilter = (WUQueryFilterBoolean) atoi(fv);
  2302. else if (!fv || !*fv)
  2303. unknownAttributes.append(getEnumText(subfmt,querySortFields));
  2304. else {
  2305. xPath.append('[').append(getEnumText(subfmt,querySortFields)).append('=');
  2306. if (fmt&WUQSFnocase)
  2307. xPath.append('?');
  2308. if (fmt&WUQSFnumeric)
  2309. xPath.append('#');
  2310. if (fmt&WUQSFwild)
  2311. xPath.append('~');
  2312. xPath.append('"').append(fv).append("\"]");
  2313. }
  2314. fv = fv + strlen(fv)+1;
  2315. }
  2316. }
  2317. if (sortorder) {
  2318. for (unsigned i=0;sortorder[i]!=WUQSFterm;i++) {
  2319. if (so.length())
  2320. so.append(',');
  2321. int fmt = sortorder[i];
  2322. if (fmt&WUQSFreverse)
  2323. so.append('-');
  2324. if (fmt&WUQSFnocase)
  2325. so.append('?');
  2326. if (fmt&WUQSFnumeric)
  2327. so.append('#');
  2328. so.append(getEnumText(fmt&0xff,querySortFields));
  2329. }
  2330. }
  2331. IArrayOf<IPropertyTree> results;
  2332. Owned<IElementsPager> elementsPager = new CQuerySetQueriesPager(querySet.get(), xPath.str(), so.length()?so.str():NULL, postFilters, unknownAttributes, _subset);
  2333. Owned<IRemoteConnection> conn=getElementsPaged(elementsPager,startoffset,maxnum,NULL,"",cachehint,results,total);
  2334. return new CConstQuerySetQueryIterator(results);
  2335. }
  2336. bool CWorkUnitFactory::isAborting(const char *wuid) const
  2337. {
  2338. VStringBuffer apath("/WorkUnitAborts/%s", wuid);
  2339. try
  2340. {
  2341. Owned<IRemoteConnection> acon = querySDS().connect(apath.str(), myProcessSession(), 0, SDS_LOCK_TIMEOUT);
  2342. if (acon)
  2343. return acon->queryRoot()->getPropInt(NULL) != 0;
  2344. }
  2345. catch (IException *E)
  2346. {
  2347. EXCLOG(E);
  2348. E->Release();
  2349. }
  2350. return false;
  2351. }
  2352. void CWorkUnitFactory::clearAborting(const char *wuid)
  2353. {
  2354. VStringBuffer apath("/WorkUnitAborts/%s", wuid);
  2355. try
  2356. {
  2357. Owned<IRemoteConnection> acon = querySDS().connect(apath.str(), myProcessSession(), RTM_LOCK_WRITE|RTM_LOCK_SUB, SDS_LOCK_TIMEOUT);
  2358. if (acon)
  2359. acon->close(true);
  2360. }
  2361. catch (IException *E)
  2362. {
  2363. EXCLOG(E);
  2364. E->Release();
  2365. }
  2366. }
  2367. static CriticalSection deleteDllLock;
  2368. static Owned<IWorkQueueThread> deleteDllWorkQ;
  2369. static void asyncRemoveDll(const char * name)
  2370. {
  2371. CriticalBlock b(deleteDllLock);
  2372. if (!deleteDllWorkQ)
  2373. deleteDllWorkQ.setown(createWorkQueueThread());
  2374. deleteDllWorkQ->post(new asyncRemoveDllWorkItem(name));
  2375. }
  2376. static void asyncRemoveFile(const char * ip, const char * name)
  2377. {
  2378. CriticalBlock b(deleteDllLock);
  2379. if (!deleteDllWorkQ)
  2380. deleteDllWorkQ.setown(createWorkQueueThread());
  2381. deleteDllWorkQ->post(new asyncRemoveRemoteFileWorkItem(ip, name));
  2382. }
  2383. class CDaliWorkUnitFactory : public CWorkUnitFactory, implements IDaliClientShutdown
  2384. {
  2385. public:
  2386. IMPLEMENT_IINTERFACE_USING(CWorkUnitFactory);
  2387. CDaliWorkUnitFactory()
  2388. {
  2389. // Assumes dali client configuration has already been done
  2390. sdsManager = &querySDS();
  2391. session = myProcessSession();
  2392. addShutdownHook(*this);
  2393. }
  2394. ~CDaliWorkUnitFactory()
  2395. {
  2396. removeShutdownHook(*this);
  2397. }
  2398. virtual unsigned validateRepository(bool fixErrors)
  2399. {
  2400. return 0;
  2401. }
  2402. virtual void deleteRepository(bool recreate)
  2403. {
  2404. Owned<IRemoteConnection> conn = sdsManager->connect("/WorkUnits", session, RTM_LOCK_WRITE, SDS_LOCK_TIMEOUT);
  2405. conn->close(true);
  2406. }
  2407. virtual void createRepository()
  2408. {
  2409. // Nothing to do
  2410. }
  2411. virtual const char *queryStoreType() const
  2412. {
  2413. return "Dali";
  2414. }
  2415. virtual CLocalWorkUnit *_createWorkUnit(const char *wuid, ISecManager *secmgr, ISecUser *secuser)
  2416. {
  2417. StringBuffer wuRoot;
  2418. getXPath(wuRoot, wuid);
  2419. IRemoteConnection *conn;
  2420. conn = sdsManager->connect(wuRoot.str(), session, RTM_LOCK_WRITE|RTM_CREATE_UNIQUE, SDS_LOCK_TIMEOUT);
  2421. conn->queryRoot()->setProp("@xmlns:xsi", "http://www.w3.org/1999/XMLSchema-instance");
  2422. conn->queryRoot()->setPropInt("@wuidVersion", WUID_VERSION);
  2423. return new CDaliWorkUnit(conn, secmgr, secuser);
  2424. }
  2425. virtual CLocalWorkUnit* _openWorkUnit(const char *wuid, bool lock, ISecManager *secmgr, ISecUser *secuser)
  2426. {
  2427. StringBuffer wuRoot;
  2428. getXPath(wuRoot, wuid);
  2429. IRemoteConnection* conn = sdsManager->connect(wuRoot.str(), session, lock ? RTM_LOCK_READ|RTM_LOCK_SUB : 0, SDS_LOCK_TIMEOUT);
  2430. if (conn)
  2431. return new CDaliWorkUnit(conn, secmgr, secuser);
  2432. else
  2433. return NULL;
  2434. }
  2435. virtual CLocalWorkUnit* _updateWorkUnit(const char *wuid, ISecManager *secmgr, ISecUser *secuser)
  2436. {
  2437. StringBuffer wuRoot;
  2438. getXPath(wuRoot, wuid);
  2439. IRemoteConnection* conn = sdsManager->connect(wuRoot.str(), session, RTM_LOCK_WRITE|RTM_LOCK_SUB, SDS_LOCK_TIMEOUT);
  2440. if (conn)
  2441. return new CDaliWorkUnit(conn, secmgr, secuser);
  2442. else
  2443. return NULL;
  2444. }
  2445. virtual IWorkUnit* getGlobalWorkUnit(ISecManager *secmgr, ISecUser *secuser)
  2446. {
  2447. // MORE - should it check security?
  2448. StringBuffer wuRoot;
  2449. getXPath(wuRoot, GLOBAL_WORKUNIT);
  2450. IRemoteConnection* conn = sdsManager->connect(wuRoot.str(), session, RTM_LOCK_WRITE|RTM_CREATE_QUERY, SDS_LOCK_TIMEOUT);
  2451. conn->queryRoot()->setProp("@xmlns:xsi", "http://www.w3.org/1999/XMLSchema-instance");
  2452. Owned<CLocalWorkUnit> cw = new CDaliWorkUnit(conn, (ISecManager *) NULL, NULL);
  2453. return &cw->lockRemote(false);
  2454. }
  2455. virtual IConstWorkUnitIterator* getWorkUnitsByOwner(const char * owner, ISecManager *secmgr, ISecUser *secuser)
  2456. {
  2457. StringBuffer path("*");
  2458. if (owner && *owner)
  2459. path.append("[@submitID=?~\"").append(owner).append("\"]");
  2460. return _getWorkUnitsByXPath(path.str(), secmgr, secuser);
  2461. }
  2462. IConstWorkUnitIterator* getScheduledWorkUnits(ISecManager *secmgr, ISecUser *secuser)
  2463. {
  2464. StringBuffer path("*");
  2465. path.append("[@state=\"").append(getEnumText(WUStateScheduled, states)).append("\"]");
  2466. return _getWorkUnitsByXPath(path.str(), secmgr, secuser);
  2467. }
  2468. virtual void clientShutdown();
  2469. virtual unsigned numWorkUnits()
  2470. {
  2471. Owned<IRemoteConnection> conn = sdsManager->connect("/WorkUnits", session, 0, SDS_LOCK_TIMEOUT);
  2472. if (!conn)
  2473. return 0;
  2474. IPropertyTree *root = conn->queryRoot();
  2475. return root->numChildren();
  2476. }
  2477. IConstWorkUnitIterator* getWorkUnitsSorted( WUSortField sortorder, // field to sort by (and flags for desc sort etc)
  2478. WUSortField *filters, // NULL or list of fields to filter on (terminated by WUSFterm)
  2479. const void *filterbuf, // (appended) string values for filters
  2480. unsigned startoffset,
  2481. unsigned maxnum,
  2482. const char *queryowner,
  2483. __int64 *cachehint,
  2484. unsigned *total,
  2485. ISecManager *secmgr,
  2486. ISecUser *secuser)
  2487. {
  2488. class CWorkUnitsPager : public CSimpleInterface, implements IElementsPager
  2489. {
  2490. StringAttr xPath;
  2491. StringAttr sortOrder;
  2492. StringAttr nameFilterLo;
  2493. StringAttr nameFilterHi;
  2494. StringArray unknownAttributes;
  2495. public:
  2496. IMPLEMENT_IINTERFACE_USING(CSimpleInterface);
  2497. CWorkUnitsPager(const char* _xPath, const char *_sortOrder, const char* _nameFilterLo, const char* _nameFilterHi, StringArray& _unknownAttributes)
  2498. : xPath(_xPath), sortOrder(_sortOrder), nameFilterLo(_nameFilterLo), nameFilterHi(_nameFilterHi)
  2499. {
  2500. ForEachItemIn(x, _unknownAttributes)
  2501. unknownAttributes.append(_unknownAttributes.item(x));
  2502. }
  2503. virtual IRemoteConnection* getElements(IArrayOf<IPropertyTree> &elements)
  2504. {
  2505. Owned<IRemoteConnection> conn = querySDS().connect("WorkUnits", myProcessSession(), 0, SDS_LOCK_TIMEOUT);
  2506. if (!conn)
  2507. return NULL;
  2508. Owned<IPropertyTreeIterator> iter = conn->getElements(xPath);
  2509. if (!iter)
  2510. return NULL;
  2511. sortElements(iter, sortOrder.get(), nameFilterLo.get(), nameFilterHi.get(), unknownAttributes, elements);
  2512. return conn.getClear();
  2513. }
  2514. };
  2515. class CScopeChecker : public CSimpleInterface, implements ISortedElementsTreeFilter
  2516. {
  2517. UniqueScopes done;
  2518. ISecManager *secmgr;
  2519. ISecUser *secuser;
  2520. CriticalSection crit;
  2521. public:
  2522. IMPLEMENT_IINTERFACE_USING(CSimpleInterface);
  2523. CScopeChecker(ISecManager *_secmgr,ISecUser *_secuser)
  2524. {
  2525. secmgr = _secmgr;
  2526. secuser = _secuser;
  2527. }
  2528. bool isOK(IPropertyTree &tree)
  2529. {
  2530. const char *scopename = tree.queryProp("@scope");
  2531. if (!scopename||!*scopename)
  2532. return true;
  2533. {
  2534. CriticalBlock block(crit);
  2535. const bool *b = done.getValue(scopename);
  2536. if (b)
  2537. return *b;
  2538. }
  2539. bool ret = checkWuScopeSecAccess(scopename,secmgr,secuser,SecAccess_Read,"iterating",false,false);
  2540. {
  2541. // conceivably could have already been checked and added, but ok.
  2542. CriticalBlock block(crit);
  2543. done.setValue(scopename,ret);
  2544. }
  2545. return ret;
  2546. }
  2547. };
  2548. Owned<ISortedElementsTreeFilter> sc = new CScopeChecker(secmgr,secuser);
  2549. StringBuffer query;
  2550. StringBuffer so;
  2551. StringAttr namefilter("*");
  2552. StringAttr namefilterlo;
  2553. StringAttr namefilterhi;
  2554. StringArray unknownAttributes;
  2555. if (filters) {
  2556. const char *fv = (const char *)filterbuf;
  2557. for (unsigned i=0;filters[i]!=WUSFterm;i++) {
  2558. int fmt = filters[i];
  2559. int subfmt = (fmt&0xff);
  2560. if (subfmt==WUSFwuid)
  2561. namefilterlo.set(fv);
  2562. else if (subfmt==WUSFwuidhigh)
  2563. namefilterhi.set(fv);
  2564. else if (subfmt==WUSFwildwuid)
  2565. namefilter.set(fv);
  2566. else if (subfmt==WUSFcustom)
  2567. query.append("[").append(fv).append("]");
  2568. else if (!fv || !*fv)
  2569. unknownAttributes.append(getEnumText(subfmt,workunitSortFields));
  2570. else {
  2571. query.append('[').append(getEnumText(subfmt,workunitSortFields)).append('=');
  2572. if (fmt&WUSFnocase)
  2573. query.append('?');
  2574. if (fmt&WUSFwild)
  2575. query.append('~');
  2576. query.append('"').append(fv).append("\"]");
  2577. }
  2578. fv = fv + strlen(fv)+1;
  2579. }
  2580. }
  2581. query.insert(0, namefilter.get());
  2582. if (sortorder)
  2583. {
  2584. if (so.length())
  2585. so.append(',');
  2586. if (sortorder & WUSFreverse)
  2587. so.append('-');
  2588. if (sortorder & WUSFnocase)
  2589. so.append('?');
  2590. if (sortorder & WUSFnumeric)
  2591. so.append('#');
  2592. so.append(getEnumText(sortorder&0xff,workunitSortFields));
  2593. }
  2594. IArrayOf<IPropertyTree> results;
  2595. Owned<IElementsPager> elementsPager = new CWorkUnitsPager(query.str(), so.length()?so.str():NULL, namefilterlo.get(), namefilterhi.get(), unknownAttributes);
  2596. Owned<IRemoteConnection> conn=getElementsPaged(elementsPager,startoffset,maxnum,secmgr?sc:NULL,queryowner,cachehint,results,total);
  2597. return new CConstWUArrayIterator(results);
  2598. }
  2599. virtual WUState waitForWorkUnit(const char * wuid, unsigned timeout, bool compiled, bool returnOnWaitState)
  2600. {
  2601. StringBuffer wuRoot;
  2602. getXPath(wuRoot, wuid);
  2603. Owned<WorkUnitWaiter> waiter = new WorkUnitWaiter(wuRoot.str());
  2604. LocalIAbortHandler abortHandler(*waiter);
  2605. WUState ret = WUStateUnknown;
  2606. Owned<IRemoteConnection> conn = sdsManager->connect(wuRoot.str(), session, 0, SDS_LOCK_TIMEOUT);
  2607. if (conn)
  2608. {
  2609. unsigned start = msTick();
  2610. loop
  2611. {
  2612. ret = (WUState) getEnum(conn->queryRoot(), "@state", states);
  2613. switch (ret)
  2614. {
  2615. case WUStateCompiled:
  2616. case WUStateUploadingFiles:
  2617. if (!compiled)
  2618. break;
  2619. // fall into
  2620. case WUStateCompleted:
  2621. case WUStateFailed:
  2622. case WUStateAborted:
  2623. waiter->unsubscribe();
  2624. return ret;
  2625. case WUStateWait:
  2626. if(returnOnWaitState)
  2627. {
  2628. waiter->unsubscribe();
  2629. return ret;
  2630. }
  2631. break;
  2632. case WUStateCompiling:
  2633. case WUStateRunning:
  2634. case WUStateDebugPaused:
  2635. case WUStateDebugRunning:
  2636. case WUStateBlocked:
  2637. case WUStateAborting:
  2638. if (queryDaliServerVersion().compare("2.1")>=0)
  2639. {
  2640. SessionId agent = conn->queryRoot()->getPropInt64("@agentSession", -1);
  2641. if((agent>0) && querySessionManager().sessionStopped(agent, 0))
  2642. {
  2643. waiter->unsubscribe();
  2644. conn->reload();
  2645. ret = (WUState) getEnum(conn->queryRoot(), "@state", states);
  2646. bool isEcl = false;
  2647. switch (ret)
  2648. {
  2649. case WUStateCompiling:
  2650. isEcl = true;
  2651. // drop into
  2652. case WUStateRunning:
  2653. case WUStateBlocked:
  2654. ret = WUStateFailed;
  2655. break;
  2656. case WUStateAborting:
  2657. ret = WUStateAborted;
  2658. break;
  2659. default:
  2660. return ret;
  2661. }
  2662. WARNLOG("_waitForWorkUnit terminated: %" I64F "d state = %d",(__int64)agent,(int)ret);
  2663. Owned<IWorkUnitFactory> factory = getWorkUnitFactory();
  2664. Owned<IWorkUnit> wu = factory->updateWorkUnit(wuid);
  2665. wu->setState(ret);
  2666. Owned<IWUException> e = wu->createException();
  2667. e->setExceptionCode(isEcl ? 1001 : 1000);
  2668. e->setExceptionMessage(isEcl ? "EclServer terminated unexpectedly" : "Workunit terminated unexpectedly");
  2669. return ret;
  2670. }
  2671. }
  2672. break;
  2673. }
  2674. unsigned waited = msTick() - start;
  2675. if (timeout==-1)
  2676. {
  2677. waiter->wait(20000); // recheck state every 20 seconds even if no timeout, in case eclagent has crashed.
  2678. if (waiter->aborted)
  2679. {
  2680. ret = WUStateUnknown; // MORE - throw an exception?
  2681. break;
  2682. }
  2683. }
  2684. else if (waited > timeout || !waiter->wait(timeout-waited))
  2685. {
  2686. ret = WUStateUnknown; // MORE - throw an exception?
  2687. break;
  2688. }
  2689. conn->reload();
  2690. }
  2691. }
  2692. waiter->unsubscribe();
  2693. return ret;
  2694. }
  2695. protected:
  2696. class WorkUnitWaiter : public CInterface, implements ISDSSubscription, implements IAbortHandler
  2697. {
  2698. Semaphore changed;
  2699. SubscriptionId change;
  2700. public:
  2701. IMPLEMENT_IINTERFACE;
  2702. WorkUnitWaiter(const char *xpath)
  2703. {
  2704. change = querySDS().subscribe(xpath, *this, false);
  2705. aborted = false;
  2706. }
  2707. ~WorkUnitWaiter()
  2708. {
  2709. assertex(change==0);
  2710. }
  2711. void notify(SubscriptionId id, const char *xpath, SDSNotifyFlags flags, unsigned valueLen, const void *valueData)
  2712. {
  2713. changed.signal();
  2714. }
  2715. bool wait(unsigned timeout)
  2716. {
  2717. return changed.wait(timeout) && !aborted;
  2718. }
  2719. bool onAbort()
  2720. {
  2721. aborted = true;
  2722. changed.signal();
  2723. return false;
  2724. }
  2725. void unsubscribe()
  2726. {
  2727. querySDS().unsubscribe(change);
  2728. change = 0;
  2729. }
  2730. bool aborted;
  2731. };
  2732. IConstWorkUnitIterator * _getWorkUnitsByXPath(const char *xpath, ISecManager *secmgr, ISecUser *secuser)
  2733. {
  2734. Owned<IRemoteConnection> conn = sdsManager->connect("/WorkUnits", session, 0, SDS_LOCK_TIMEOUT);
  2735. if (conn)
  2736. {
  2737. CDaliVersion serverVersionNeeded("3.2");
  2738. Owned<IPropertyTreeIterator> iter(queryDaliServerVersion().compare(serverVersionNeeded) < 0 ?
  2739. conn->queryRoot()->getElements(xpath) :
  2740. conn->getElements(xpath));
  2741. return createSecureConstWUIterator(iter.getClear(), secmgr, secuser);
  2742. }
  2743. else
  2744. return NULL;
  2745. }
  2746. ISDSManager *sdsManager;
  2747. SessionId session;
  2748. };
  2749. extern WORKUNIT_API IConstWorkUnitIterator *createSecureConstWUIterator(IConstWorkUnitIterator *iter, ISecManager *secmgr, ISecUser *secuser)
  2750. {
  2751. if (secmgr)
  2752. return new CSecureConstWUIterator(iter, secmgr, secuser);
  2753. else
  2754. return iter;
  2755. }
  2756. extern WORKUNIT_API IConstWorkUnitIterator *createSecureConstWUIterator(IPropertyTreeIterator *iter, ISecManager *secmgr, ISecUser *secuser)
  2757. {
  2758. if (secmgr)
  2759. return new CSecureConstWUIterator(new CConstWUIterator(iter), secmgr, secuser);
  2760. else
  2761. return new CConstWUIterator(iter);
  2762. }
  2763. static CriticalSection factoryCrit;
  2764. static Owned<ILoadedDllEntry> workunitServerPlugin; // NOTE - unload AFTER the factory is released!
  2765. static Owned<IWorkUnitFactory> factory;
  2766. void CDaliWorkUnitFactory::clientShutdown()
  2767. {
  2768. CriticalBlock b(factoryCrit);
  2769. factory.clear();
  2770. }
  2771. void clientShutdownWorkUnit()
  2772. {
  2773. CriticalBlock b(factoryCrit);
  2774. factory.clear();
  2775. }
  2776. extern WORKUNIT_API void setWorkUnitFactory(IWorkUnitFactory * _factory)
  2777. {
  2778. CriticalBlock b(factoryCrit);
  2779. factory.setown(_factory);
  2780. }
  2781. extern WORKUNIT_API IWorkUnitFactory * getWorkUnitFactory()
  2782. {
  2783. if (!factory)
  2784. {
  2785. CriticalBlock b(factoryCrit);
  2786. if (!factory) // NOTE - this "double test" paradigm is not guaranteed threadsafe on modern systems/compilers - I think in this instance that is harmless even in the (extremely) unlikely event that it resulted in the setown being called twice.
  2787. {
  2788. const char *forceEnv = getenv("FORCE_DALI_WORKUNITS");
  2789. bool forceDali = forceEnv && !strieq(forceEnv, "off") && !strieq(forceEnv, "0");
  2790. Owned<IRemoteConnection> conn = querySDS().connect("/Environment/Software/WorkUnitsServer", myProcessSession(), 0, SDS_LOCK_TIMEOUT);
  2791. // MORE - arguably should be looking in the config section that corresponds to the dali we connected to. If you want to allow some dalis to be configured to use a WU server and others not.
  2792. if (conn && !forceDali)
  2793. {
  2794. const IPropertyTree *ptree = conn->queryRoot();
  2795. const char *pluginName = ptree->queryProp("@plugin");
  2796. if (!pluginName)
  2797. throw makeStringException(WUERR_WorkunitPluginError, "WorkUnitsServer information missing plugin name");
  2798. workunitServerPlugin.setown(createDllEntry(pluginName, false, NULL));
  2799. if (!workunitServerPlugin)
  2800. throw makeStringExceptionV(WUERR_WorkunitPluginError, "WorkUnitsServer: failed to load plugin %s", pluginName);
  2801. WorkUnitFactoryFactory pf = (WorkUnitFactoryFactory) workunitServerPlugin->getEntry("createWorkUnitFactory");
  2802. if (!pf)
  2803. throw makeStringExceptionV(WUERR_WorkunitPluginError, "WorkUnitsServer: function createWorkUnitFactory not found in plugin %s", pluginName);
  2804. factory.setown(pf(ptree));
  2805. if (!factory)
  2806. throw makeStringExceptionV(WUERR_WorkunitPluginError, "WorkUnitsServer: createWorkUnitFactory returned NULL in plugin %s", pluginName);
  2807. }
  2808. else
  2809. factory.setown(new CDaliWorkUnitFactory());
  2810. }
  2811. }
  2812. return factory.getLink();
  2813. }
  2814. // A SecureWorkUnitFactory allows the security params to be supplied once to the factory rather than being supplied to each call.
  2815. // They can still be supplied if you want...
  2816. class CSecureWorkUnitFactory : public CInterface, implements IWorkUnitFactory
  2817. {
  2818. public:
  2819. IMPLEMENT_IINTERFACE;
  2820. CSecureWorkUnitFactory(IWorkUnitFactory *_baseFactory, ISecManager *_secMgr, ISecUser *_secUser)
  2821. : baseFactory(_baseFactory), defaultSecMgr(_secMgr), defaultSecUser(_secUser)
  2822. {
  2823. }
  2824. virtual unsigned validateRepository(bool fix)
  2825. {
  2826. return baseFactory->validateRepository(fix);
  2827. }
  2828. virtual void deleteRepository(bool recreate)
  2829. {
  2830. return baseFactory->deleteRepository(recreate);
  2831. }
  2832. virtual void createRepository()
  2833. {
  2834. return baseFactory->createRepository();
  2835. }
  2836. virtual const char *queryStoreType() const
  2837. {
  2838. return baseFactory->queryStoreType();
  2839. }
  2840. virtual IWorkUnit* createNamedWorkUnit(const char *wuid, const char *app, const char *user, ISecManager *secMgr, ISecUser *secUser)
  2841. {
  2842. if (!secMgr) secMgr = defaultSecMgr.get();
  2843. if (!secUser) secUser = defaultSecUser.get();
  2844. return baseFactory->createNamedWorkUnit(wuid, app, user, secMgr, secUser);
  2845. }
  2846. virtual IWorkUnit* createWorkUnit(const char *app, const char *user, ISecManager *secMgr, ISecUser *secUser)
  2847. {
  2848. if (!secMgr) secMgr = defaultSecMgr.get();
  2849. if (!secUser) secUser = defaultSecUser.get();
  2850. return baseFactory->createWorkUnit(app, user, secMgr, secUser);
  2851. }
  2852. virtual bool deleteWorkUnit(const char * wuid, ISecManager *secMgr, ISecUser *secUser)
  2853. {
  2854. if (!secMgr) secMgr = defaultSecMgr.get();
  2855. if (!secUser) secUser = defaultSecUser.get();
  2856. return baseFactory->deleteWorkUnit(wuid, secMgr, secUser);
  2857. }
  2858. virtual IConstWorkUnit* openWorkUnit(const char *wuid, bool lock, ISecManager *secMgr, ISecUser *secUser)
  2859. {
  2860. if (!secMgr) secMgr = defaultSecMgr.get();
  2861. if (!secUser) secUser = defaultSecUser.get();
  2862. return baseFactory->openWorkUnit(wuid, lock, secMgr, secUser);
  2863. }
  2864. virtual IWorkUnit* updateWorkUnit(const char *wuid, ISecManager *secMgr, ISecUser *secUser)
  2865. {
  2866. if (!secMgr) secMgr = defaultSecMgr.get();
  2867. if (!secUser) secUser = defaultSecUser.get();
  2868. return baseFactory->updateWorkUnit(wuid, secMgr, secUser);
  2869. }
  2870. virtual IWorkUnit * getGlobalWorkUnit(ISecManager *secMgr, ISecUser *secUser)
  2871. {
  2872. if (!secMgr) secMgr = defaultSecMgr.get();
  2873. if (!secUser) secUser = defaultSecUser.get();
  2874. return baseFactory->getGlobalWorkUnit(secMgr, secUser);
  2875. }
  2876. virtual IConstWorkUnitIterator * getWorkUnitsByOwner(const char * owner, ISecManager *secMgr, ISecUser *secUser)
  2877. {
  2878. if (!secMgr) secMgr = defaultSecMgr.get();
  2879. if (!secUser) secUser = defaultSecUser.get();
  2880. return baseFactory->getWorkUnitsByOwner(owner, secMgr, secUser);
  2881. }
  2882. virtual IConstWorkUnitIterator * getScheduledWorkUnits(ISecManager *secMgr, ISecUser *secUser)
  2883. {
  2884. if (!secMgr) secMgr = defaultSecMgr.get();
  2885. if (!secUser) secUser = defaultSecUser.get();
  2886. return baseFactory->getScheduledWorkUnits(secMgr, secUser);
  2887. }
  2888. virtual void descheduleAllWorkUnits(ISecManager *secMgr, ISecUser *secUser)
  2889. {
  2890. if (!secMgr) secMgr = defaultSecMgr.get();
  2891. if (!secUser) secUser = defaultSecUser.get();
  2892. baseFactory->descheduleAllWorkUnits(secMgr, secUser);
  2893. }
  2894. virtual int setTracingLevel(int newLevel)
  2895. {
  2896. return baseFactory->setTracingLevel(newLevel);
  2897. }
  2898. virtual IConstWorkUnitIterator* getWorkUnitsSorted( WUSortField sortorder, // field to sort by
  2899. WUSortField *filters, // NULL or list of fields to filter on (terminated by WUSFterm)
  2900. const void *filterbuf, // (appended) string values for filters
  2901. unsigned startoffset,
  2902. unsigned maxnum,
  2903. const char *queryowner,
  2904. __int64 *cachehint,
  2905. unsigned *total,
  2906. ISecManager *secMgr, ISecUser *secUser)
  2907. {
  2908. if (!secMgr) secMgr = defaultSecMgr.get();
  2909. if (!secUser) secUser = defaultSecUser.get();
  2910. return baseFactory->getWorkUnitsSorted(sortorder,filters,filterbuf,startoffset,maxnum,queryowner,cachehint, total, secMgr, secUser);
  2911. }
  2912. virtual IConstQuerySetQueryIterator* getQuerySetQueriesSorted( WUQuerySortField *sortorder,
  2913. WUQuerySortField *filters,
  2914. const void *filterbuf,
  2915. unsigned startoffset,
  2916. unsigned maxnum,
  2917. __int64 *cachehint,
  2918. unsigned *total,
  2919. const MapStringTo<bool> *subset)
  2920. {
  2921. // MORE - why no security?
  2922. return baseFactory->getQuerySetQueriesSorted(sortorder,filters,filterbuf,startoffset,maxnum,cachehint,total,subset);
  2923. }
  2924. virtual unsigned numWorkUnits()
  2925. {
  2926. return baseFactory->numWorkUnits();
  2927. }
  2928. virtual bool isAborting(const char *wuid) const
  2929. {
  2930. return baseFactory->isAborting(wuid);
  2931. }
  2932. virtual void clearAborting(const char *wuid)
  2933. {
  2934. baseFactory->clearAborting(wuid);
  2935. }
  2936. virtual WUState waitForWorkUnit(const char * wuid, unsigned timeout, bool compiled, bool returnOnWaitState)
  2937. {
  2938. return baseFactory->waitForWorkUnit(wuid, timeout, compiled, returnOnWaitState);
  2939. }
  2940. private:
  2941. Owned<IWorkUnitFactory> baseFactory;
  2942. Linked<ISecManager> defaultSecMgr;
  2943. Linked<ISecUser> defaultSecUser;
  2944. };
  2945. extern WORKUNIT_API IWorkUnitFactory * getWorkUnitFactory(ISecManager *secmgr, ISecUser *secuser)
  2946. {
  2947. if (secmgr && secuser)
  2948. return new CSecureWorkUnitFactory(getWorkUnitFactory(), secmgr, secuser);
  2949. else
  2950. return getWorkUnitFactory();
  2951. }
  2952. //==========================================================================================
  2953. class CStringPTreeIterator : public CInterface, implements IStringIterator
  2954. {
  2955. Owned<IPropertyTreeIterator> it;
  2956. public:
  2957. IMPLEMENT_IINTERFACE;
  2958. CStringPTreeIterator(IPropertyTreeIterator *p) : it(p) {};
  2959. virtual bool first() { return it->first(); }
  2960. virtual bool next() { return it->next(); }
  2961. virtual bool isValid() { return it->isValid(); }
  2962. virtual IStringVal & str(IStringVal &s) { s.set(it->query().queryProp(NULL)); return s; }
  2963. };
  2964. class CStringPTreeTagIterator : public CInterface, implements IStringIterator
  2965. {
  2966. Owned<IPropertyTreeIterator> it;
  2967. public:
  2968. IMPLEMENT_IINTERFACE;
  2969. CStringPTreeTagIterator(IPropertyTreeIterator *p) : it(p) {};
  2970. virtual bool first() { return it->first(); }
  2971. virtual bool next() { return it->next(); }
  2972. virtual bool isValid() { return it->isValid(); }
  2973. virtual IStringVal & str(IStringVal &s) { s.set(it->query().queryName()); return s; }
  2974. };
  2975. class CStringPTreeAttrIterator : public CInterface, implements IStringIterator
  2976. {
  2977. Owned<IPropertyTreeIterator> it;
  2978. StringAttr name;
  2979. public:
  2980. IMPLEMENT_IINTERFACE;
  2981. CStringPTreeAttrIterator(IPropertyTreeIterator *p, const char *_name) : it(p), name(_name) {};
  2982. virtual bool first() { return it->first(); }
  2983. virtual bool next() { return it->next(); }
  2984. virtual bool isValid() { return it->isValid(); }
  2985. virtual IStringVal & str(IStringVal &s) { s.set(it->query().queryProp(name)); return s; }
  2986. };
  2987. //==========================================================================================
  2988. CLocalWorkUnit::CLocalWorkUnit(ISecManager *secmgr, ISecUser *secuser)
  2989. {
  2990. clearCached(false);
  2991. secMgr.set(secmgr);
  2992. secUser.set(secuser);
  2993. workflowIteratorCached = false;
  2994. resultsCached = false;
  2995. temporariesCached = false;
  2996. variablesCached = false;
  2997. exceptionsCached = false;
  2998. pluginsCached = false;
  2999. librariesCached = false;
  3000. activitiesCached = false;
  3001. webServicesInfoCached = false;
  3002. roxieQueryInfoCached = false;
  3003. }
  3004. void CLocalWorkUnit::clearCached(bool clearTree)
  3005. {
  3006. query.clear();
  3007. webServicesInfo.clear();
  3008. roxieQueryInfo.clear();
  3009. workflowIterator.clear();
  3010. cachedGraphs.clear();
  3011. graphs.kill();
  3012. results.kill();
  3013. variables.kill();
  3014. plugins.kill();
  3015. libraries.kill();
  3016. exceptions.kill();
  3017. temporaries.kill();
  3018. statistics.kill();
  3019. appvalues.kill();
  3020. if (clearTree)
  3021. p.clear();
  3022. workflowIteratorCached = false;
  3023. resultsCached = false;
  3024. temporariesCached = false;
  3025. variablesCached = false;
  3026. exceptionsCached = false;
  3027. pluginsCached = false;
  3028. librariesCached = false;
  3029. activitiesCached = false;
  3030. webServicesInfoCached = false;
  3031. roxieQueryInfoCached = false;
  3032. }
  3033. void CLocalWorkUnit::loadPTree(IPropertyTree *ptree)
  3034. {
  3035. clearCached(false);
  3036. p.setown(ptree);
  3037. }
  3038. void CLocalWorkUnit::beforeDispose()
  3039. {
  3040. try
  3041. {
  3042. unsubscribe();
  3043. clearCached(true);
  3044. userDesc.clear();
  3045. secMgr.clear();
  3046. secUser.clear();
  3047. }
  3048. catch (IException *E) { LOG(MCexception(E, MSGCLS_warning), E, "Exception during ~CLocalWorkUnit"); E->Release(); }
  3049. }
  3050. void CLocalWorkUnit::cleanupAndDelete(bool deldll, bool deleteOwned, const StringArray *deleteExclusions)
  3051. {
  3052. MTIME_SECTION(queryActiveTimer(), "WUDELETE cleanupAndDelete total");
  3053. // Delete any related things in SDS etc that might otherwise be forgotten
  3054. if (p->getPropBool("@protected", false))
  3055. throw MakeStringException(WUERR_WorkunitProtected, "%s: Workunit is protected",p->queryName());
  3056. switch (getState())
  3057. {
  3058. case WUStateAborted:
  3059. case WUStateCompleted:
  3060. case WUStateFailed:
  3061. case WUStateArchived:
  3062. break;
  3063. case WUStateCompiled:
  3064. if (getAction()==WUActionRun || getAction()==WUActionUnknown)
  3065. throw MakeStringException(WUERR_WorkunitActive, "%s: Workunit is active",p->queryName());
  3066. break;
  3067. case WUStateWait:
  3068. throw MakeStringException(WUERR_WorkunitScheduled, "%s: Workunit is scheduled",p->queryName());
  3069. default:
  3070. throw MakeStringException(WUERR_WorkunitActive, "%s: Workunit is active",p->queryName());
  3071. break;
  3072. }
  3073. if (getIsQueryService())
  3074. {
  3075. Owned<IPropertyTree> registry = getQueryRegistryRoot();
  3076. if (registry)
  3077. {
  3078. VStringBuffer xpath("QuerySet/Query[@wuid='%s']", p->queryName());
  3079. if (registry->hasProp(xpath.str()))
  3080. throw MakeStringException(WUERR_WorkunitPublished, "%s: Workunit is published",p->queryName());
  3081. }
  3082. }
  3083. try
  3084. {
  3085. if (deldll && !p->getPropBool("@isClone", false))
  3086. {
  3087. Owned<IConstWUQuery> q = getQuery();
  3088. if (q)
  3089. {
  3090. Owned<IConstWUAssociatedFileIterator> iter = &q->getAssociatedFiles();
  3091. SCMStringBuffer name;
  3092. ForEach(*iter)
  3093. {
  3094. IConstWUAssociatedFile & cur = iter->query();
  3095. cur.getNameTail(name);
  3096. if (!deleteExclusions || (NotFound == deleteExclusions->find(name.str())))
  3097. {
  3098. Owned<IDllEntry> entry = queryDllServer().getEntry(name.str());
  3099. if (entry.get())
  3100. asyncRemoveDll(name.str());
  3101. else
  3102. {
  3103. SCMStringBuffer ip, localPath;
  3104. cur.getName(localPath);
  3105. cur.getIp(ip);
  3106. asyncRemoveFile(ip.str(), localPath.str());
  3107. }
  3108. }
  3109. }
  3110. }
  3111. }
  3112. factory->clearAborting(queryWuid());
  3113. deleteTempFiles(NULL, deleteOwned, true); // all, any remaining.
  3114. }
  3115. catch(IException *E)
  3116. {
  3117. StringBuffer s;
  3118. LOG(MCexception(E, MSGCLS_warning), E, s.append("Exception during cleanupAndDelete: ").append(p->queryName()).str());
  3119. E->Release();
  3120. }
  3121. catch (...)
  3122. {
  3123. WARNLOG("Unknown exception during cleanupAndDelete: %s", p->queryName());
  3124. }
  3125. CConstGraphProgress::deleteWuidProgress(p->queryName());
  3126. }
  3127. void CLocalWorkUnit::setTimeScheduled(const IJlibDateTime &val)
  3128. {
  3129. SCMStringBuffer strval;
  3130. val.getGmtString(strval);
  3131. p->setProp("@timescheduled",strval.str());
  3132. }
  3133. IJlibDateTime & CLocalWorkUnit::getTimeScheduled(IJlibDateTime &val) const
  3134. {
  3135. StringBuffer str;
  3136. p->getProp("@timescheduled",str);
  3137. if(str.length())
  3138. val.setGmtString(str.str());
  3139. return val;
  3140. }
  3141. bool modifyAndWriteWorkUnitXML(char const * wuid, StringBuffer & buf, StringBuffer & extra, IFileIO * fileio)
  3142. {
  3143. // kludge in extra chunks of XML such as GraphProgress and GeneratedDlls
  3144. if(extra.length())
  3145. {
  3146. size32_t l = (size32_t)strlen(wuid);
  3147. size32_t p = buf.length()-l-4; // bit of a kludge
  3148. assertex(memcmp(buf.str()+p+2,wuid,l)==0);
  3149. StringAttr tail(buf.str()+p);
  3150. buf.setLength(p);
  3151. buf.append(extra);
  3152. buf.append(tail);
  3153. }
  3154. return (fileio->write(0,buf.length(),buf.str()) == buf.length());
  3155. }
  3156. bool CLocalWorkUnit::archiveWorkUnit(const char *base,bool del,bool ignoredllerrors,bool deleteOwned)
  3157. {
  3158. CriticalBlock block(crit);
  3159. StringBuffer path(base);
  3160. if (!p)
  3161. return false;
  3162. const char *wuid = p->queryName();
  3163. if (!wuid||!*wuid)
  3164. return false;
  3165. addPathSepChar(path).append(wuid).append(".xml");
  3166. Owned<IFile> file = createIFile(path.str());
  3167. if (!file)
  3168. return false;
  3169. Owned<IFileIO> fileio = file->open(IFOcreate);
  3170. if (!fileio)
  3171. return false;
  3172. StringBuffer buf;
  3173. exportWorkUnitToXML(this, buf, false, false, true);
  3174. StringBuffer extraWorkUnitXML;
  3175. StringBuffer xpath("/GraphProgress/");
  3176. xpath.append(wuid);
  3177. Owned<IRemoteConnection> conn = querySDS().connect(xpath.str(), myProcessSession(), RTM_LOCK_WRITE, SDS_LOCK_TIMEOUT);
  3178. if (conn)
  3179. {
  3180. Owned<IPropertyTree> tmp = createPTree("GraphProgress");
  3181. mergePTree(tmp,conn->queryRoot());
  3182. toXML(tmp,extraWorkUnitXML,1,XML_Format);
  3183. conn->close();
  3184. }
  3185. Owned<IConstWUQuery> q = getQuery();
  3186. if (!q)
  3187. {
  3188. if (!modifyAndWriteWorkUnitXML(wuid, buf, extraWorkUnitXML, fileio))
  3189. return false;
  3190. if (del)
  3191. {
  3192. if (getState()==WUStateUnknown)
  3193. setState(WUStateArchived); // to allow delete
  3194. cleanupAndDelete(false,deleteOwned); // no query, may as well delete
  3195. }
  3196. return false;
  3197. }
  3198. StringArray deleteExclusions; // associated files not to delete, added if failure to copy
  3199. Owned<IConstWUAssociatedFileIterator> iter = &q->getAssociatedFiles();
  3200. Owned<IPropertyTree> generatedDlls = createPTree("GeneratedDlls");
  3201. ForEach(*iter)
  3202. {
  3203. IConstWUAssociatedFile & cur = iter->query();
  3204. SCMStringBuffer name;
  3205. cur.getNameTail(name);
  3206. if (name.length())
  3207. {
  3208. Owned<IDllEntry> entry = queryDllServer().getEntry(name.str());
  3209. SCMStringBuffer curPath, curIp;
  3210. cur.getName(curPath);
  3211. cur.getIp(curIp);
  3212. SocketEndpoint curEp(curIp.str());
  3213. RemoteFilename curRfn;
  3214. curRfn.setPath(curEp, curPath.str());
  3215. StringBuffer dst(base);
  3216. addPathSepChar(dst);
  3217. curRfn.getTail(dst);
  3218. Owned<IFile> dstFile = createIFile(dst.str());
  3219. if (entry.get())
  3220. {
  3221. Owned<IException> exception;
  3222. Owned<IDllLocation> loc;
  3223. Owned<IPropertyTree> generatedDllBranch = createPTree();
  3224. generatedDllBranch->setProp("@name", entry->queryName());
  3225. generatedDllBranch->setProp("@kind", entry->queryKind());
  3226. try
  3227. {
  3228. loc.setown(entry->getBestLocation()); //throws exception if no readable locations
  3229. }
  3230. catch(IException * e)
  3231. {
  3232. exception.setown(e);
  3233. loc.setown(entry->getBestLocationCandidate()); //this will be closest of the unreadable locations
  3234. }
  3235. RemoteFilename filename;
  3236. loc->getDllFilename(filename);
  3237. if (!exception)
  3238. {
  3239. Owned<IFile> srcfile = createIFile(filename);
  3240. try
  3241. {
  3242. if (dstFile->exists())
  3243. {
  3244. if (streq(srcfile->queryFilename(), dstFile->queryFilename()))
  3245. deleteExclusions.append(name.str()); // restored workunit, referencing archive location for query dll (no longer true post HPCC-11191 fix)
  3246. // still want to delete if already archived but there are source file copies
  3247. }
  3248. else
  3249. copyFile(dstFile, srcfile);
  3250. }
  3251. catch(IException * e)
  3252. {
  3253. exception.setown(e);
  3254. }
  3255. }
  3256. if (exception)
  3257. {
  3258. if (ignoredllerrors)
  3259. {
  3260. EXCLOG(exception.get(), "archiveWorkUnit (copying associated file)");
  3261. //copy failed, so don't delete the registred dll files
  3262. deleteExclusions.append(name.str());
  3263. }
  3264. else
  3265. throw exception.getClear();
  3266. }
  3267. // Record Associated path to restore back to
  3268. StringBuffer restorePath;
  3269. curRfn.getRemotePath(restorePath);
  3270. generatedDllBranch->setProp("@location", restorePath.str());
  3271. generatedDlls->addPropTree("GeneratedDll", generatedDllBranch.getClear());
  3272. }
  3273. else // no generated dll entry
  3274. {
  3275. Owned<IFile> srcFile = createIFile(curRfn);
  3276. try
  3277. {
  3278. copyFile(dstFile, srcFile);
  3279. }
  3280. catch (IException *e)
  3281. {
  3282. VStringBuffer msg("Failed to archive associated file '%s' to destination '%s'", srcFile->queryFilename(), dstFile->queryFilename());
  3283. EXCLOG(e, msg.str());
  3284. e->Release();
  3285. deleteExclusions.append(name.str());
  3286. }
  3287. }
  3288. }
  3289. }
  3290. iter.clear();
  3291. if (generatedDlls->numChildren())
  3292. toXML(generatedDlls, extraWorkUnitXML, 1, XML_Format);
  3293. if (!modifyAndWriteWorkUnitXML(wuid, buf, extraWorkUnitXML, fileio))
  3294. return false;
  3295. if (del)
  3296. {
  3297. //setState(WUStateArchived); // this isn't useful as about to delete it!
  3298. q.clear();
  3299. cleanupAndDelete(true, deleteOwned, &deleteExclusions);
  3300. }
  3301. return true;
  3302. }
  3303. void CLocalWorkUnit::packWorkUnit(bool pack)
  3304. {
  3305. // only packs Graph info currently
  3306. CriticalBlock block(crit);
  3307. if (!p)
  3308. return;
  3309. const char *wuid = p->queryName();
  3310. if (!wuid||!*wuid)
  3311. return;
  3312. if (pack) {
  3313. if (!p->hasProp("PackedGraphs")) {
  3314. cachedGraphs.clear();
  3315. IPropertyTree *t = p->queryPropTree("Graphs");
  3316. if (t) {
  3317. MemoryBuffer buf;
  3318. t->serialize(buf);
  3319. p->setPropBin("PackedGraphs",buf.length(),buf.bufferBase());
  3320. p->removeTree(t);
  3321. }
  3322. }
  3323. }
  3324. else {
  3325. ensureGraphsUnpacked();
  3326. }
  3327. CConstGraphProgress::packProgress(wuid,pack);
  3328. }
  3329. IPropertyTree * pruneBranch(IPropertyTree * from, char const * xpath)
  3330. {
  3331. Owned<IPropertyTree> ret;
  3332. IPropertyTree * branch = from->queryPropTree(xpath);
  3333. if(branch) {
  3334. ret.setown(createPTreeFromIPT(branch));
  3335. from->removeTree(branch);
  3336. }
  3337. return ret.getClear();
  3338. }
  3339. bool restoreWorkUnit(const char *base,const char *wuid)
  3340. {
  3341. StringBuffer path(base);
  3342. if (!wuid||!*wuid)
  3343. return false;
  3344. addPathSepChar(path).append(wuid).append(".xml");
  3345. Owned<IFile> file = createIFile(path.str());
  3346. if (!file)
  3347. return false;
  3348. Owned<IFileIO> fileio = file->open(IFOread);
  3349. if (!fileio)
  3350. return false;
  3351. Owned<IPropertyTree> pt = createPTree(*fileio);
  3352. if (!pt)
  3353. return false;
  3354. CDateTime dt;
  3355. dt.setNow();
  3356. StringBuffer dts;
  3357. dt.getString(dts);
  3358. pt->setProp("@restoredDate", dts.str());
  3359. VStringBuffer xpath("/WorkUnits/%s", wuid);
  3360. Owned<IRemoteConnection> conn = querySDS().connect(xpath.str(), myProcessSession(), RTM_LOCK_WRITE|RTM_CREATE_QUERY, SDS_LOCK_TIMEOUT);
  3361. if (!conn)
  3362. {
  3363. ERRLOG("restoreWorkUnit could not create to %s", xpath.str());
  3364. return false;
  3365. }
  3366. IPropertyTree *root = conn->queryRoot();
  3367. if (root->hasChildren())
  3368. {
  3369. ERRLOG("restoreWorkUnit WUID %s already exists", wuid);
  3370. return false;
  3371. }
  3372. Owned<IPropertyTree> gprogress = pruneBranch(pt, "GraphProgress[1]");
  3373. Owned<IPropertyTree> generatedDlls = pruneBranch(pt, "GeneratedDlls[1]");
  3374. Owned<IPropertyTree> associatedFiles;
  3375. IPropertyTree *srcAssociated = pt->queryPropTree("Query/Associated");
  3376. if (srcAssociated)
  3377. associatedFiles.setown(createPTreeFromIPT(srcAssociated));
  3378. root->setPropTree(NULL, pt.getClear());
  3379. conn.clear();
  3380. // now kludge back GraphProgress and GeneratedDlls
  3381. if (gprogress)
  3382. {
  3383. VStringBuffer xpath("/GraphProgress/%s", wuid);
  3384. conn.setown(querySDS().connect(xpath, myProcessSession(), RTM_LOCK_WRITE|RTM_CREATE_QUERY, SDS_LOCK_TIMEOUT));
  3385. if (conn)
  3386. {
  3387. IPropertyTree *groot = conn->queryRoot();
  3388. if (groot->hasChildren())
  3389. WARNLOG("restoreWorkUnit WUID %s graphprogress already exists, replacing",wuid);
  3390. groot->setPropTree(NULL, gprogress.getClear());
  3391. }
  3392. }
  3393. if (generatedDlls)
  3394. {
  3395. Owned<IPropertyTreeIterator> dlls = generatedDlls->getElements("GeneratedDll");
  3396. for(dlls->first(); dlls->isValid(); dlls->next())
  3397. {
  3398. IPropertyTree & dll = dlls->query();
  3399. char const * name = dll.queryProp("@name");
  3400. char const * kind = dll.queryProp("@kind");
  3401. char const * location = dll.queryProp("@location");
  3402. Owned<IDllEntry> got = queryDllServer().getEntry(name);
  3403. if (!got)
  3404. {
  3405. RemoteFilename dstRfn;
  3406. dstRfn.setRemotePath(location);
  3407. StringBuffer srcPath(base);
  3408. addPathSepChar(srcPath);
  3409. dstRfn.getTail(srcPath);
  3410. OwnedIFile srcFile = createIFile(srcPath);
  3411. OwnedIFile dstFile = createIFile(dstRfn);
  3412. copyFile(dstFile, srcFile);
  3413. queryDllServer().registerDll(name, kind, location);
  3414. }
  3415. }
  3416. }
  3417. if (associatedFiles)
  3418. {
  3419. Owned<IPropertyTreeIterator> associated = associatedFiles->getElements("*");
  3420. ForEach(*associated)
  3421. {
  3422. IPropertyTree &file = associated->query();
  3423. const char *filename = file.queryProp("@filename");
  3424. SocketEndpoint ep(file.queryProp("@ip"));
  3425. RemoteFilename rfn;
  3426. rfn.setPath(ep, filename);
  3427. OwnedIFile dstFile = createIFile(rfn);
  3428. StringBuffer srcPath(base), name;
  3429. addPathSepChar(srcPath);
  3430. rfn.getTail(name);
  3431. srcPath.append(name);
  3432. if (generatedDlls)
  3433. {
  3434. VStringBuffer gDllPath("GeneratedDll[@name=\"%s\"]", name.str());
  3435. if (generatedDlls->hasProp(gDllPath))
  3436. continue; // generated dlls handled separately - see above
  3437. }
  3438. OwnedIFile srcFile = createIFile(srcPath);
  3439. if (srcFile->exists())
  3440. {
  3441. try
  3442. {
  3443. copyFile(dstFile, srcFile);
  3444. }
  3445. catch (IException *e)
  3446. {
  3447. VStringBuffer msg("Failed to restore associated file '%s' to destination '%s'", srcFile->queryFilename(), dstFile->queryFilename());
  3448. EXCLOG(e, msg.str());
  3449. e->Release();
  3450. }
  3451. }
  3452. }
  3453. }
  3454. return true;
  3455. }
  3456. void CLocalWorkUnit::loadXML(const char *xml)
  3457. {
  3458. CriticalBlock block(crit);
  3459. clearCached(true);
  3460. assertex(xml);
  3461. p.setown(createPTreeFromXMLString(xml));
  3462. }
  3463. void CLocalWorkUnit::serialize(MemoryBuffer &tgt)
  3464. {
  3465. CriticalBlock block(crit);
  3466. StringBuffer x;
  3467. tgt.append(exportWorkUnitToXML(this, x, false, false, false).str());
  3468. }
  3469. void CLocalWorkUnit::deserialize(MemoryBuffer &src)
  3470. {
  3471. CriticalBlock block(crit);
  3472. StringAttr value;
  3473. src.read(value);
  3474. loadXML(value);
  3475. }
  3476. void CLocalWorkUnit::requestAbort()
  3477. {
  3478. CriticalBlock block(crit);
  3479. abortWorkUnit(p->queryName());
  3480. }
  3481. void CLocalWorkUnit::subscribe(WUSubscribeOptions options)
  3482. {
  3483. }
  3484. void CLocalWorkUnit::unlockRemote()
  3485. {
  3486. CriticalBlock block(crit);
  3487. locked.unlock();
  3488. if (IsShared()) // Is this right? Doesn't feel right! Commit on last unlock would seem smarter
  3489. {
  3490. _unlockRemote();
  3491. }
  3492. }
  3493. IWorkUnit &CLocalWorkUnit::lockRemote(bool commit)
  3494. {
  3495. if (secMgr)
  3496. checkWuSecAccess(*this, secMgr.get(), secUser.get(), SecAccess_Write, "write lock", true, true);
  3497. locked.lock();
  3498. CriticalBlock block(crit);
  3499. if (commit)
  3500. {
  3501. try
  3502. {
  3503. _lockRemote();
  3504. }
  3505. catch (IException *E)
  3506. {
  3507. StringBuffer s;
  3508. PrintLog("Failed to get write lock on workunit: %s", E->errorMessage(s).str());
  3509. locked.unlock();
  3510. throw;
  3511. }
  3512. }
  3513. return *new CLockedWorkUnit(LINK(this));
  3514. }
  3515. void CLocalWorkUnit::commit()
  3516. {
  3517. // Nothing to do if not backed by a persistent store
  3518. }
  3519. IWorkUnit& CLocalWorkUnit::lock()
  3520. {
  3521. return lockRemote(true);
  3522. }
  3523. const char *CLocalWorkUnit::queryWuid() const
  3524. {
  3525. CriticalBlock block(crit);
  3526. return p->queryName();
  3527. }
  3528. unsigned CLocalWorkUnit::getDebugAgentListenerPort() const
  3529. {
  3530. CriticalBlock block(crit);
  3531. return p->getPropInt("@DebugListenerPort", 0);
  3532. }
  3533. void CLocalWorkUnit::setDebugAgentListenerPort(unsigned port)
  3534. {
  3535. CriticalBlock block(crit);
  3536. p->setPropInt("@DebugListenerPort", port);
  3537. }
  3538. IStringVal& CLocalWorkUnit::getDebugAgentListenerIP(IStringVal &ip) const
  3539. {
  3540. CriticalBlock block(crit);
  3541. ip.set(p->queryProp("@DebugListenerIP"));
  3542. return ip;
  3543. }
  3544. void CLocalWorkUnit::setDebugAgentListenerIP(const char * ip)
  3545. {
  3546. CriticalBlock block(crit);
  3547. p->setProp("@DebugListenerIP", ip);
  3548. }
  3549. IStringVal& CLocalWorkUnit::getSecurityToken(IStringVal &str) const
  3550. {
  3551. CriticalBlock block(crit);
  3552. str.set(p->queryProp("@token"));
  3553. return str;
  3554. }
  3555. void CLocalWorkUnit::setSecurityToken(const char *value)
  3556. {
  3557. CriticalBlock block(crit);
  3558. p->setProp("@token", value);
  3559. }
  3560. bool CLocalWorkUnit::getRunningGraph(IStringVal &graphName, WUGraphIDType &subId) const
  3561. {
  3562. return CConstGraphProgress::getRunningGraph(p->queryName(), graphName, subId);
  3563. }
  3564. void CLocalWorkUnit::setJobName(const char *value)
  3565. {
  3566. CriticalBlock block(crit);
  3567. p->setProp("@jobName", value);
  3568. }
  3569. const char *CLocalWorkUnit::queryJobName() const
  3570. {
  3571. CriticalBlock block(crit);
  3572. const char *ret = p->queryProp("@jobName");
  3573. if (!ret)
  3574. ret = "";
  3575. return ret;
  3576. }
  3577. void CLocalWorkUnit::setClusterName(const char *value)
  3578. {
  3579. CriticalBlock block(crit);
  3580. p->setProp("@clusterName", value);
  3581. }
  3582. const char *CLocalWorkUnit::queryClusterName() const
  3583. {
  3584. CriticalBlock block(crit);
  3585. const char *ret = p->queryProp("@clusterName");
  3586. if (!ret)
  3587. ret = "";
  3588. return ret;
  3589. }
  3590. void CLocalWorkUnit::setAllowedClusters(const char *value)
  3591. {
  3592. setDebugValue("allowedclusters",value, true);
  3593. }
  3594. IStringVal& CLocalWorkUnit::getAllowedClusters(IStringVal &str) const
  3595. {
  3596. CriticalBlock block(crit);
  3597. getDebugValue("allowedclusters",str);
  3598. if (str.length()!=0)
  3599. return str;
  3600. str.set(p->queryProp("@clusterName"));
  3601. return str;
  3602. }
  3603. void CLocalWorkUnit::setAllowAutoQueueSwitch(bool val)
  3604. {
  3605. setDebugValueInt("allowautoqueueswitch",val?1:0,true);
  3606. }
  3607. bool CLocalWorkUnit::getAllowAutoQueueSwitch() const
  3608. {
  3609. CriticalBlock block(crit);
  3610. return getDebugValueBool("allowautoqueueswitch",false);
  3611. }
  3612. void CLocalWorkUnit::setLibraryInformation(const char * name, unsigned interfaceHash, unsigned definitionHash)
  3613. {
  3614. StringBuffer suffix;
  3615. if (name && *name)
  3616. setApplicationValue("LibraryModule", "name", name, true);
  3617. setApplicationValueInt("LibraryModule", "interfaceHash", interfaceHash, true);
  3618. setApplicationValueInt("LibraryModule", "definitionHash", definitionHash, true);
  3619. setApplicationValue("LibraryModule", "platform", appendLibrarySuffix(suffix).str(), true);
  3620. }
  3621. void CLocalWorkUnit::remoteCheckAccess(IUserDescriptor *user, bool writeaccess) const
  3622. {
  3623. unsigned auditflags = DALI_LDAP_AUDIT_REPORT|DALI_LDAP_READ_WANTED;
  3624. if (writeaccess)
  3625. auditflags |= DALI_LDAP_WRITE_WANTED;
  3626. int perm = 255;
  3627. const char *scopename = p->queryProp("@scope");
  3628. if (scopename&&*scopename) {
  3629. if (!user)
  3630. user = queryUserDescriptor();
  3631. perm = querySessionManager().getPermissionsLDAP("workunit",scopename,user,auditflags);
  3632. if (perm<0) {
  3633. if (perm==-1)
  3634. perm = 255;
  3635. else
  3636. perm = 0;
  3637. }
  3638. }
  3639. if (!HASREADPERMISSION(perm))
  3640. throw MakeStringException(WUERR_WorkunitAccessDenied, "Read access denied for workunit %s", queryWuid());
  3641. if (writeaccess && !HASWRITEPERMISSION(perm))
  3642. throw MakeStringException(WUERR_WorkunitAccessDenied, "Write access denied for workunit %s", queryWuid());
  3643. }
  3644. void CLocalWorkUnit::setUser(const char * value)
  3645. {
  3646. CriticalBlock block(crit);
  3647. p->setProp("@submitID", value);
  3648. }
  3649. const char *CLocalWorkUnit::queryUser() const
  3650. {
  3651. CriticalBlock block(crit);
  3652. const char *ret = p->queryProp("@submitID");
  3653. if (!ret)
  3654. ret = "";
  3655. return ret;
  3656. }
  3657. void CLocalWorkUnit::setWuScope(const char * value)
  3658. {
  3659. if (value && *value)
  3660. {
  3661. if (checkWuScopeSecAccess(value, secMgr.get(), secUser.get(), SecAccess_Write, "Change Scope", true, true))
  3662. {
  3663. CriticalBlock block(crit);
  3664. p->setProp("@scope", value);
  3665. }
  3666. }
  3667. }
  3668. const char *CLocalWorkUnit::queryWuScope() const
  3669. {
  3670. CriticalBlock block(crit);
  3671. const char *ret = p->queryProp("@scope");
  3672. if (!ret)
  3673. ret = "";
  3674. return ret;
  3675. }
  3676. mapEnums priorityClasses[] = {
  3677. { PriorityClassUnknown, "unknown" },
  3678. { PriorityClassLow, "low" },
  3679. { PriorityClassNormal, "normal" },
  3680. { PriorityClassHigh, "high" },
  3681. { PriorityClassSize, NULL },
  3682. };
  3683. void CLocalWorkUnit::setPriority(WUPriorityClass cls)
  3684. {
  3685. CriticalBlock block(crit);
  3686. setEnum(p, "@priorityClass", cls, priorityClasses);
  3687. }
  3688. WUPriorityClass CLocalWorkUnit::getPriority() const
  3689. {
  3690. CriticalBlock block(crit);
  3691. return (WUPriorityClass) getEnum(p, "@priorityClass", priorityClasses);
  3692. }
  3693. void CLocalWorkUnit::setState(WUState value)
  3694. {
  3695. if (value==WUStateAborted || value==WUStatePaused || value==WUStateCompleted || value==WUStateFailed || value==WUStateSubmitted || value==WUStateWait)
  3696. {
  3697. if (factory)
  3698. factory->clearAborting(queryWuid());
  3699. }
  3700. CriticalBlock block(crit);
  3701. setEnum(p, "@state", value, states);
  3702. if (getDebugValueBool("monitorWorkunit", false))
  3703. {
  3704. switch(value)
  3705. {
  3706. case WUStateAborted:
  3707. FLLOG(MCoperatorWarning, "Workunit %s aborted", p->queryName());
  3708. break;
  3709. case WUStateCompleted:
  3710. FLLOG(MCoperatorProgress, "Workunit %s completed", p->queryName());
  3711. break;
  3712. case WUStateFailed:
  3713. FLLOG(MCoperatorProgress, "Workunit %s failed", p->queryName());
  3714. break;
  3715. }
  3716. }
  3717. p->removeProp("@stateEx");
  3718. }
  3719. void CLocalWorkUnit::setStateEx(const char * text)
  3720. {
  3721. CriticalBlock block(crit);
  3722. p->setProp("@stateEx", text);
  3723. }
  3724. void CLocalWorkUnit::setAgentSession(__int64 sessionId)
  3725. {
  3726. CriticalBlock block(crit);
  3727. p->setPropInt64("@agentSession", sessionId);
  3728. }
  3729. bool CLocalWorkUnit::aborting() const
  3730. {
  3731. return false;
  3732. }
  3733. bool CLocalWorkUnit::getIsQueryService() const
  3734. {
  3735. CriticalBlock block(crit);
  3736. return p->getPropBool("@isQueryService", false);
  3737. }
  3738. void CLocalWorkUnit::setIsQueryService(bool value)
  3739. {
  3740. CriticalBlock block(crit);
  3741. p->setPropBool("@isQueryService", value);
  3742. }
  3743. void CLocalWorkUnit::checkAgentRunning(WUState & state)
  3744. {
  3745. if (queryDaliServerVersion().compare("2.1")<0)
  3746. return;
  3747. switch(state)
  3748. {
  3749. case WUStateRunning:
  3750. case WUStateDebugPaused:
  3751. case WUStateDebugRunning:
  3752. case WUStateBlocked:
  3753. case WUStateAborting:
  3754. case WUStateCompiling:
  3755. case WUStatePaused:
  3756. {
  3757. SessionId agent = getAgentSession();
  3758. if((agent>0) && querySessionManager().sessionStopped(agent, 0))
  3759. {
  3760. forceReload();
  3761. state = (WUState) getEnum(p, "@state", states);
  3762. bool isecl=state==WUStateCompiling;
  3763. if (aborting())
  3764. state = WUStateAborted;
  3765. else if (state==WUStateRunning || state==WUStatePaused || state==WUStateDebugPaused || state==WUStateDebugRunning || state==WUStateBlocked || state==WUStateCompiling)
  3766. state = WUStateFailed;
  3767. else
  3768. return;
  3769. WARNLOG("checkAgentRunning terminated: %" I64F "d state = %d",(__int64)agent,(int)state);
  3770. Owned<IWorkUnit> w = &lock();
  3771. w->setState(state);
  3772. Owned<IWUException> e = w->createException();
  3773. WUAction action = w->getAction();
  3774. switch (action)
  3775. {
  3776. case WUActionPause:
  3777. case WUActionPauseNow:
  3778. case WUActionResume:
  3779. w->setAction(WUActionUnknown);
  3780. }
  3781. if(isecl)
  3782. {
  3783. e->setExceptionCode(1001);
  3784. e->setExceptionMessage("EclServer terminated unexpectedly");
  3785. }
  3786. else
  3787. {
  3788. e->setExceptionCode(1000);
  3789. e->setExceptionMessage("Workunit terminated unexpectedly");
  3790. }
  3791. }
  3792. }
  3793. }
  3794. }
  3795. WUState CLocalWorkUnit::getState() const
  3796. {
  3797. CriticalBlock block(crit);
  3798. WUState state = (WUState) getEnum(p, "@state", states);
  3799. switch (state)
  3800. {
  3801. case WUStateRunning:
  3802. case WUStateDebugPaused:
  3803. case WUStateDebugRunning:
  3804. case WUStateBlocked:
  3805. case WUStateCompiling:
  3806. if (aborting())
  3807. state = WUStateAborting;
  3808. break;
  3809. case WUStateSubmitted:
  3810. if (aborting())
  3811. state = WUStateAborted;
  3812. break;
  3813. }
  3814. const_cast<CLocalWorkUnit *>(this)->checkAgentRunning(state); //need const_cast as will change state if agent has died
  3815. return state;
  3816. }
  3817. IStringVal& CLocalWorkUnit::getStateEx(IStringVal & str) const
  3818. {
  3819. CriticalBlock block(crit);
  3820. str.set(p->queryProp("@stateEx"));
  3821. return str;
  3822. }
  3823. __int64 CLocalWorkUnit::getAgentSession() const
  3824. {
  3825. CriticalBlock block(crit);
  3826. return p->getPropInt64("@agentSession", -1);
  3827. }
  3828. unsigned CLocalWorkUnit::getAgentPID() const
  3829. {
  3830. CriticalBlock block(crit);
  3831. return p->getPropInt("@agentPID", -1);
  3832. }
  3833. const char * CLocalWorkUnit::queryStateDesc() const
  3834. {
  3835. // MORE - not sure about this - may prefer a separate interface
  3836. CriticalBlock block(crit);
  3837. try
  3838. {
  3839. return getEnumText(getState(), states);
  3840. }
  3841. catch (...)
  3842. {
  3843. return "???";
  3844. }
  3845. }
  3846. mapEnums actions[] = {
  3847. { WUActionUnknown, "unknown" },
  3848. { WUActionCompile, "compile" },
  3849. { WUActionCheck, "check" },
  3850. { WUActionRun, "run" },
  3851. { WUActionExecuteExisting, "execute" },
  3852. { WUActionPause, "pause" },
  3853. { WUActionPauseNow, "pausenow" },
  3854. { WUActionResume, "resume" },
  3855. { WUActionSize, NULL },
  3856. };
  3857. void CLocalWorkUnit::setAction(WUAction value)
  3858. {
  3859. CriticalBlock block(crit);
  3860. setEnum(p, "Action", value, actions);
  3861. }
  3862. WUAction CLocalWorkUnit::getAction() const
  3863. {
  3864. CriticalBlock block(crit);
  3865. return (WUAction) getEnum(p, "Action", actions);
  3866. }
  3867. IStringVal& CLocalWorkUnit::getActionEx(IStringVal & str) const
  3868. {
  3869. CriticalBlock block(crit);
  3870. str.set(p->queryProp("Action"));
  3871. return str;
  3872. }
  3873. IStringVal& CLocalWorkUnit::getApplicationValue(const char *app, const char *propname, IStringVal &str) const
  3874. {
  3875. CriticalBlock block(crit);
  3876. StringBuffer prop("Application/");
  3877. prop.append(app).append('/').append(propname);
  3878. str.set(p->queryProp(prop.str()));
  3879. return str;
  3880. }
  3881. int CLocalWorkUnit::getApplicationValueInt(const char *app, const char *propname, int defVal) const
  3882. {
  3883. CriticalBlock block(crit);
  3884. StringBuffer prop("Application/");
  3885. prop.append(app).append('/').append(propname);
  3886. return p->getPropInt(prop.str(), defVal);
  3887. }
  3888. IConstWUAppValueIterator& CLocalWorkUnit::getApplicationValues() const
  3889. {
  3890. CriticalBlock block(crit);
  3891. appvalues.load(p,"Application/*");
  3892. return *new CArrayIteratorOf<IConstWUAppValue,IConstWUAppValueIterator> (appvalues, 0, (IConstWorkUnit *) this);
  3893. }
  3894. void CLocalWorkUnit::setApplicationValue(const char *app, const char *propname, const char *value, bool overwrite)
  3895. {
  3896. CriticalBlock block(crit);
  3897. StringBuffer prop("Application/");
  3898. prop.append(app).append('/').append(propname);
  3899. if (overwrite || !p->hasProp(prop.str()))
  3900. {
  3901. // MORE - not sure these lines should be needed....
  3902. StringBuffer sp;
  3903. p->setProp(sp.append("Application").str(), "");
  3904. p->setProp(sp.append('/').append(app).str(), "");
  3905. p->setProp(prop.str(), value);
  3906. }
  3907. }
  3908. void CLocalWorkUnit::setApplicationValueInt(const char *app, const char *propname, int value, bool overwrite)
  3909. {
  3910. CriticalBlock block(crit);
  3911. StringBuffer prop("Application/");
  3912. prop.append(app).append('/').append(propname);
  3913. if (overwrite || !p->hasProp(prop.str()))
  3914. {
  3915. // MORE - not sure these lines should be needed....
  3916. StringBuffer sp;
  3917. p->setProp(sp.append("Application").str(), "");
  3918. p->setProp(sp.append('/').append(app).str(), "");
  3919. p->setPropInt(prop.str(), value);
  3920. }
  3921. }
  3922. void CLocalWorkUnit::setPriorityLevel(int level)
  3923. {
  3924. CriticalBlock block(crit);
  3925. p->setPropInt("PriorityFlag", level);
  3926. }
  3927. int CLocalWorkUnit::getPriorityLevel() const
  3928. {
  3929. CriticalBlock block(crit);
  3930. return p->getPropInt("PriorityFlag");
  3931. }
  3932. int calcPriorityValue(const IPropertyTree * p)
  3933. {
  3934. int priority = p->getPropInt("PriorityFlag");
  3935. switch((WUPriorityClass) getEnum(p, "@priorityClass", priorityClasses))
  3936. {
  3937. case PriorityClassLow:
  3938. priority -= 100;
  3939. break;
  3940. case PriorityClassHigh:
  3941. priority += 100;
  3942. break;
  3943. }
  3944. return priority;
  3945. }
  3946. int CLocalWorkUnit::getPriorityValue() const
  3947. {
  3948. CriticalBlock block(crit);
  3949. return calcPriorityValue(p);
  3950. }
  3951. void CLocalWorkUnit::setRescheduleFlag(bool value)
  3952. {
  3953. CriticalBlock block(crit);
  3954. p->setPropInt("RescheduleFlag", (int) value);
  3955. }
  3956. bool CLocalWorkUnit::getRescheduleFlag() const
  3957. {
  3958. CriticalBlock block(crit);
  3959. return p->getPropInt("RescheduleFlag") != 0;
  3960. }
  3961. class NullIStringIterator : public CInterface, extends IStringIterator
  3962. {
  3963. public:
  3964. IMPLEMENT_IINTERFACE;
  3965. bool first() { return false; }
  3966. bool next() { return false; }
  3967. bool isValid() { return false; }
  3968. IStringVal & str(IStringVal & str) { return str; }
  3969. };
  3970. ClusterType getClusterType(const char * platform, ClusterType dft)
  3971. {
  3972. if (stricmp(platform, "thor") == 0)
  3973. return ThorLCRCluster;
  3974. if (stricmp(platform, "thorlcr") == 0)
  3975. return ThorLCRCluster;
  3976. if (stricmp(platform, "hthor") == 0)
  3977. return HThorCluster;
  3978. if (stricmp(platform, "roxie") == 0)
  3979. return RoxieCluster;
  3980. return dft;
  3981. }
  3982. const char *clusterTypeString(ClusterType clusterType, bool lcrSensitive)
  3983. {
  3984. switch (clusterType)
  3985. {
  3986. case ThorLCRCluster:
  3987. if (lcrSensitive)
  3988. return "thorlcr";
  3989. return "thor";
  3990. case RoxieCluster:
  3991. return "roxie";
  3992. case HThorCluster:
  3993. return "hthor";
  3994. }
  3995. throwUnexpected();
  3996. }
  3997. IPropertyTree *queryRoxieProcessTree(IPropertyTree *environment, const char *process)
  3998. {
  3999. if (!process || !*process)
  4000. return NULL;
  4001. VStringBuffer xpath("Software/RoxieCluster[@name=\"%s\"]", process);
  4002. return environment->queryPropTree(xpath.str());
  4003. }
  4004. void getRoxieProcessServers(IPropertyTree *roxie, SocketEndpointArray &endpoints)
  4005. {
  4006. if (!roxie)
  4007. return;
  4008. Owned<IPropertyTreeIterator> servers = roxie->getElements("RoxieServerProcess");
  4009. ForEach(*servers)
  4010. {
  4011. IPropertyTree &server = servers->query();
  4012. const char *netAddress = server.queryProp("@netAddress");
  4013. if (netAddress && *netAddress)
  4014. {
  4015. SocketEndpoint ep(netAddress, server.getPropInt("@port", 9876));
  4016. endpoints.append(ep);
  4017. }
  4018. }
  4019. }
  4020. void getRoxieProcessServers(const char *process, SocketEndpointArray &servers)
  4021. {
  4022. Owned<IEnvironmentFactory> factory = getEnvironmentFactory();
  4023. Owned<IConstEnvironment> env = factory->openEnvironment();
  4024. if (!env)
  4025. return;
  4026. Owned<IPropertyTree> root = &env->getPTree();
  4027. getRoxieProcessServers(queryRoxieProcessTree(root, process), servers);
  4028. }
  4029. class CEnvironmentClusterInfo: public CInterface, implements IConstWUClusterInfo
  4030. {
  4031. StringAttr name;
  4032. StringAttr serverQueue;
  4033. StringAttr agentQueue;
  4034. StringAttr roxieProcess;
  4035. SocketEndpointArray roxieServers;
  4036. StringAttr thorQueue;
  4037. StringArray thorProcesses;
  4038. StringArray primaryThorProcesses;
  4039. StringAttr prefix;
  4040. StringAttr ldapUser;
  4041. StringBuffer ldapPassword;
  4042. ClusterType platform;
  4043. unsigned clusterWidth;
  4044. unsigned roxieRedundancy;
  4045. unsigned channelsPerNode;
  4046. int roxieReplicateOffset;
  4047. public:
  4048. IMPLEMENT_IINTERFACE;
  4049. CEnvironmentClusterInfo(const char *_name, const char *_prefix, IPropertyTree *agent, IArrayOf<IPropertyTree> &thors, IPropertyTree *roxie)
  4050. : name(_name), prefix(_prefix), roxieRedundancy(0), channelsPerNode(0), roxieReplicateOffset(1)
  4051. {
  4052. StringBuffer queue;
  4053. if (thors.ordinality())
  4054. {
  4055. thorQueue.set(getClusterThorQueueName(queue.clear(), name));
  4056. clusterWidth = 0;
  4057. bool isMultiThor = (thors.length() > 1);
  4058. ForEachItemIn(i,thors)
  4059. {
  4060. IPropertyTree &thor = thors.item(i);
  4061. const char* thorName = thor.queryProp("@name");
  4062. thorProcesses.append(thorName);
  4063. if (!isMultiThor)
  4064. primaryThorProcesses.append(thorName);
  4065. else
  4066. {
  4067. const char *nodeGroup = thor.queryProp("@nodeGroup");
  4068. if (!nodeGroup || strieq(nodeGroup, thorName))
  4069. primaryThorProcesses.append(thorName);
  4070. }
  4071. unsigned nodes = thor.getCount("ThorSlaveProcess");
  4072. if (!nodes)
  4073. throw MakeStringException(WUERR_MismatchClusterSize,"CEnvironmentClusterInfo: Thor cluster can not have 0 slave processes");
  4074. unsigned ts = nodes * thor.getPropInt("@slavesPerNode", 1);
  4075. if (clusterWidth && (ts!=clusterWidth))
  4076. throw MakeStringException(WUERR_MismatchClusterSize,"CEnvironmentClusterInfo: mismatched thor sizes in cluster");
  4077. clusterWidth = ts;
  4078. bool islcr = !thor.getPropBool("@Legacy");
  4079. if (!islcr)
  4080. throw MakeStringException(WUERR_MismatchThorType,"CEnvironmentClusterInfo: Legacy Thor no longer supported");
  4081. }
  4082. platform = ThorLCRCluster;
  4083. }
  4084. else if (roxie)
  4085. {
  4086. roxieProcess.set(roxie->queryProp("@name"));
  4087. platform = RoxieCluster;
  4088. getRoxieProcessServers(roxie, roxieServers);
  4089. clusterWidth = roxieServers.length();
  4090. ldapUser.set(roxie->queryProp("@ldapUser"));
  4091. StringBuffer encPassword = roxie->queryProp("@ldapPassword");
  4092. if (encPassword.length())
  4093. decrypt(ldapPassword, encPassword);
  4094. const char *redundancyMode = roxie->queryProp("@slaveConfig");
  4095. if (redundancyMode && *redundancyMode)
  4096. {
  4097. unsigned dataCopies = roxie->getPropInt("@numDataCopies", 1);
  4098. if (strieq(redundancyMode, "overloaded"))
  4099. channelsPerNode = roxie->getPropInt("@channelsPernode", 1);
  4100. else if (strieq(redundancyMode, "full redundancy"))
  4101. {
  4102. roxieRedundancy = dataCopies-1;
  4103. roxieReplicateOffset = 0;
  4104. }
  4105. else if (strieq(redundancyMode, "cyclic redundancy"))
  4106. {
  4107. roxieRedundancy = dataCopies-1;
  4108. channelsPerNode = dataCopies;
  4109. roxieReplicateOffset = roxie->getPropInt("@cyclicOffset", 1);
  4110. }
  4111. }
  4112. }
  4113. else
  4114. {
  4115. clusterWidth = 1;
  4116. platform = HThorCluster;
  4117. }
  4118. if (agent)
  4119. {
  4120. assertex(!roxie);
  4121. agentQueue.set(getClusterEclAgentQueueName(queue.clear(), name));
  4122. }
  4123. else if (roxie)
  4124. agentQueue.set(getClusterRoxieQueueName(queue.clear(), name));
  4125. // MORE - does this need to be conditional?
  4126. serverQueue.set(getClusterEclCCServerQueueName(queue.clear(), name));
  4127. }
  4128. IStringVal & getName(IStringVal & str) const
  4129. {
  4130. str.set(name.get());
  4131. return str;
  4132. }
  4133. IStringVal & getScope(IStringVal & str) const
  4134. {
  4135. str.set(prefix.get());
  4136. return str;
  4137. }
  4138. IStringVal & getAgentQueue(IStringVal & str) const
  4139. {
  4140. str.set(agentQueue);
  4141. return str;
  4142. }
  4143. virtual IStringVal & getServerQueue(IStringVal & str) const
  4144. {
  4145. str.set(serverQueue);
  4146. return str;
  4147. }
  4148. IStringVal & getThorQueue(IStringVal & str) const
  4149. {
  4150. str.set(thorQueue);
  4151. return str;
  4152. }
  4153. unsigned getSize() const
  4154. {
  4155. return clusterWidth;
  4156. }
  4157. virtual ClusterType getPlatform() const
  4158. {
  4159. return platform;
  4160. }
  4161. IStringVal & getRoxieProcess(IStringVal & str) const
  4162. {
  4163. str.set(roxieProcess.get());
  4164. return str;
  4165. }
  4166. const StringArray & getThorProcesses() const
  4167. {
  4168. return thorProcesses;
  4169. }
  4170. const StringArray & getPrimaryThorProcesses() const
  4171. {
  4172. return primaryThorProcesses;
  4173. }
  4174. const SocketEndpointArray & getRoxieServers() const
  4175. {
  4176. return roxieServers;
  4177. }
  4178. unsigned getRoxieRedundancy() const
  4179. {
  4180. return roxieRedundancy;
  4181. }
  4182. unsigned getChannelsPerNode() const
  4183. {
  4184. return channelsPerNode;
  4185. }
  4186. int getRoxieReplicateOffset() const
  4187. {
  4188. return roxieReplicateOffset;
  4189. }
  4190. const char *getLdapUser() const
  4191. {
  4192. return ldapUser.get();
  4193. }
  4194. virtual const char *getLdapPassword() const
  4195. {
  4196. return ldapPassword.str();
  4197. }
  4198. };
  4199. IStringVal &getProcessQueueNames(IStringVal &ret, const char *process, const char *type, const char *suffix)
  4200. {
  4201. if (process)
  4202. {
  4203. Owned<IEnvironmentFactory> factory = getEnvironmentFactory();
  4204. Owned<IConstEnvironment> env = factory->openEnvironment();
  4205. if (env)
  4206. {
  4207. Owned<IPropertyTree> root = &env->getPTree();
  4208. StringBuffer queueNames;
  4209. StringBuffer xpath;
  4210. xpath.appendf("%s[@process=\"%s\"]", type, process);
  4211. Owned<IPropertyTreeIterator> targets = root->getElements("Software/Topology/Cluster");
  4212. ForEach(*targets)
  4213. {
  4214. IPropertyTree &target = targets->query();
  4215. if (target.hasProp(xpath))
  4216. {
  4217. if (queueNames.length())
  4218. queueNames.append(',');
  4219. queueNames.append(target.queryProp("@name")).append(suffix);
  4220. }
  4221. }
  4222. ret.set(queueNames);
  4223. }
  4224. }
  4225. return ret;
  4226. }
  4227. #define ROXIE_QUEUE_EXT ".roxie"
  4228. #define THOR_QUEUE_EXT ".thor"
  4229. #define ECLCCSERVER_QUEUE_EXT ".eclserver"
  4230. #define ECLSERVER_QUEUE_EXT ECLCCSERVER_QUEUE_EXT
  4231. #define ECLSCHEDULER_QUEUE_EXT ".eclscheduler"
  4232. #define ECLAGENT_QUEUE_EXT ".agent"
  4233. extern WORKUNIT_API IStringVal &getEclCCServerQueueNames(IStringVal &ret, const char *process)
  4234. {
  4235. return getProcessQueueNames(ret, process, "EclCCServerProcess", ECLCCSERVER_QUEUE_EXT);
  4236. }
  4237. extern WORKUNIT_API IStringVal &getEclServerQueueNames(IStringVal &ret, const char *process)
  4238. {
  4239. return getProcessQueueNames(ret, process, "EclServerProcess", ECLSERVER_QUEUE_EXT); // shares queue name with EclCCServer
  4240. }
  4241. extern WORKUNIT_API IStringVal &getEclSchedulerQueueNames(IStringVal &ret, const char *process)
  4242. {
  4243. return getProcessQueueNames(ret, process, "EclSchedulerProcess", ECLSCHEDULER_QUEUE_EXT); // Shares deployment/config with EclCCServer
  4244. }
  4245. extern WORKUNIT_API IStringVal &getAgentQueueNames(IStringVal &ret, const char *process)
  4246. {
  4247. return getProcessQueueNames(ret, process, "EclAgentProcess", ECLAGENT_QUEUE_EXT);
  4248. }
  4249. extern WORKUNIT_API IStringVal &getRoxieQueueNames(IStringVal &ret, const char *process)
  4250. {
  4251. return getProcessQueueNames(ret, process, "RoxieCluster", ROXIE_QUEUE_EXT);
  4252. }
  4253. extern WORKUNIT_API IStringVal &getThorQueueNames(IStringVal &ret, const char *process)
  4254. {
  4255. return getProcessQueueNames(ret, process, "ThorCluster", THOR_QUEUE_EXT);
  4256. }
  4257. extern WORKUNIT_API StringBuffer &getClusterThorQueueName(StringBuffer &ret, const char *cluster)
  4258. {
  4259. return ret.append(cluster).append(THOR_QUEUE_EXT);
  4260. }
  4261. extern WORKUNIT_API StringBuffer &getClusterThorGroupName(StringBuffer &ret, const char *cluster)
  4262. {
  4263. Owned<IEnvironmentFactory> factory = getEnvironmentFactory();
  4264. Owned<IConstEnvironment> env = factory->openEnvironment();
  4265. if (env)
  4266. {
  4267. Owned<IPropertyTree> root = &env->getPTree();
  4268. StringBuffer path;
  4269. path.append("Software/ThorCluster[@name=\"").append(cluster).append("\"]");
  4270. IPropertyTree * child = root->queryPropTree(path);
  4271. if (child)
  4272. getClusterGroupName(*child, ret);
  4273. }
  4274. return ret;
  4275. }
  4276. extern WORKUNIT_API StringBuffer &getClusterRoxieQueueName(StringBuffer &ret, const char *cluster)
  4277. {
  4278. return ret.append(cluster).append(ROXIE_QUEUE_EXT);
  4279. }
  4280. extern WORKUNIT_API StringBuffer &getClusterEclCCServerQueueName(StringBuffer &ret, const char *cluster)
  4281. {
  4282. return ret.append(cluster).append(ECLCCSERVER_QUEUE_EXT);
  4283. }
  4284. extern WORKUNIT_API StringBuffer &getClusterEclServerQueueName(StringBuffer &ret, const char *cluster)
  4285. {
  4286. return ret.append(cluster).append(ECLSERVER_QUEUE_EXT);
  4287. }
  4288. extern WORKUNIT_API StringBuffer &getClusterEclAgentQueueName(StringBuffer &ret, const char *cluster)
  4289. {
  4290. return ret.append(cluster).append(ECLAGENT_QUEUE_EXT);
  4291. }
  4292. extern WORKUNIT_API IStringIterator *getTargetClusters(const char *processType, const char *processName)
  4293. {
  4294. Owned<CStringArrayIterator> ret = new CStringArrayIterator;
  4295. Owned<IEnvironmentFactory> factory = getEnvironmentFactory();
  4296. Owned<IConstEnvironment> env = factory->openEnvironment();
  4297. if (env)
  4298. {
  4299. Owned<IPropertyTree> root = &env->getPTree();
  4300. StringBuffer xpath;
  4301. xpath.appendf("%s", processType ? processType : "*");
  4302. if (processName && *processName)
  4303. xpath.appendf("[@process=\"%s\"]", processName);
  4304. Owned<IPropertyTreeIterator> targets = root->getElements("Software/Topology/Cluster");
  4305. ForEach(*targets)
  4306. {
  4307. IPropertyTree &target = targets->query();
  4308. if (target.hasProp(xpath))
  4309. {
  4310. ret->append(target.queryProp("@name"));
  4311. }
  4312. }
  4313. }
  4314. return ret.getClear();
  4315. }
  4316. extern WORKUNIT_API bool isProcessCluster(const char *process)
  4317. {
  4318. if (!process || !*process)
  4319. return false;
  4320. Owned<IEnvironmentFactory> factory = getEnvironmentFactory();
  4321. Owned<IConstEnvironment> env = factory->openEnvironment();
  4322. if (!env)
  4323. return false;
  4324. Owned<IPropertyTree> root = &env->getPTree();
  4325. VStringBuffer xpath("Software/*Cluster[@name=\"%s\"]", process);
  4326. return root->hasProp(xpath.str());
  4327. }
  4328. extern WORKUNIT_API bool isProcessCluster(const char *remoteDali, const char *process)
  4329. {
  4330. if (!remoteDali || !*remoteDali)
  4331. return isProcessCluster(process);
  4332. if (!process || !*process)
  4333. return false;
  4334. Owned<INode> remote = createINode(remoteDali, 7070);
  4335. if (!remote)
  4336. return false;
  4337. //Cannot use getEnvironmentFactory() since it is using a remotedali
  4338. VStringBuffer xpath("Environment/Software/*Cluster[@name=\"%s\"]/@name", process);
  4339. try
  4340. {
  4341. Owned<IPropertyTreeIterator> clusters = querySDS().getElementsRaw(xpath, remote, 1000*60*1);
  4342. return clusters->first();
  4343. }
  4344. catch (IException *E)
  4345. {
  4346. StringBuffer msg;
  4347. E->errorMessage(msg);
  4348. DBGLOG("Exception validating cluster %s/%s: %s", remoteDali, xpath.str(), msg.str());
  4349. E->Release();
  4350. }
  4351. return true;
  4352. }
  4353. IConstWUClusterInfo* getTargetClusterInfo(IPropertyTree *environment, IPropertyTree *cluster)
  4354. {
  4355. const char *clustname = cluster->queryProp("@name");
  4356. // MORE - at the moment configenf specifies eclagent and thor queues by (in effect) placing an 'example' thor or eclagent in the topology
  4357. // that uses the queue that will be used.
  4358. // We should and I hope will change that, at which point the code below gets simpler
  4359. StringBuffer prefix(cluster->queryProp("@prefix"));
  4360. prefix.toLowerCase();
  4361. StringBuffer xpath;
  4362. StringBuffer querySetName;
  4363. IPropertyTree *agent = NULL;
  4364. const char *agentName = cluster->queryProp("EclAgentProcess/@process");
  4365. if (agentName)
  4366. {
  4367. xpath.clear().appendf("Software/EclAgentProcess[@name=\"%s\"]", agentName);
  4368. agent = environment->queryPropTree(xpath.str());
  4369. }
  4370. Owned<IPropertyTreeIterator> ti = cluster->getElements("ThorCluster");
  4371. IArrayOf<IPropertyTree> thors;
  4372. ForEach(*ti)
  4373. {
  4374. const char *thorName = ti->query().queryProp("@process");
  4375. if (thorName)
  4376. {
  4377. xpath.clear().appendf("Software/ThorCluster[@name=\"%s\"]", thorName);
  4378. thors.append(*environment->getPropTree(xpath.str()));
  4379. }
  4380. }
  4381. const char *roxieName = cluster->queryProp("RoxieCluster/@process");
  4382. return new CEnvironmentClusterInfo(clustname, prefix, agent, thors, queryRoxieProcessTree(environment, roxieName));
  4383. }
  4384. IPropertyTree* getTopologyCluster(Owned<IPropertyTree> &envRoot, const char *clustname)
  4385. {
  4386. if (!clustname || !*clustname)
  4387. return NULL;
  4388. Owned<IEnvironmentFactory> factory = getEnvironmentFactory();
  4389. Owned<IConstEnvironment> env = factory->openEnvironment();
  4390. if (!env)
  4391. return NULL;
  4392. envRoot.setown(&env->getPTree());
  4393. StringBuffer xpath;
  4394. xpath.appendf("Software/Topology/Cluster[@name=\"%s\"]", clustname);
  4395. return envRoot->getPropTree(xpath.str());
  4396. }
  4397. bool validateTargetClusterName(const char *clustname)
  4398. {
  4399. Owned<IPropertyTree> envRoot;
  4400. Owned<IPropertyTree> cluster = getTopologyCluster(envRoot, clustname);
  4401. return (cluster.get()!=NULL);
  4402. }
  4403. IConstWUClusterInfo* getTargetClusterInfo(const char *clustname)
  4404. {
  4405. Owned<IPropertyTree> envRoot;
  4406. Owned<IPropertyTree> cluster = getTopologyCluster(envRoot, clustname);
  4407. if (!cluster)
  4408. return NULL;
  4409. return getTargetClusterInfo(envRoot, cluster);
  4410. }
  4411. unsigned getEnvironmentClusterInfo(CConstWUClusterInfoArray &clusters)
  4412. {
  4413. Owned<IEnvironmentFactory> factory = getEnvironmentFactory();
  4414. Owned<IConstEnvironment> env = factory->openEnvironment();
  4415. if (!env)
  4416. return 0;
  4417. Owned<IPropertyTree> root = &env->getPTree();
  4418. return getEnvironmentClusterInfo(root, clusters);
  4419. }
  4420. unsigned getEnvironmentClusterInfo(IPropertyTree* environmentRoot, CConstWUClusterInfoArray &clusters)
  4421. {
  4422. if (!environmentRoot)
  4423. return 0;
  4424. Owned<IPropertyTreeIterator> clusterIter = environmentRoot->getElements("Software/Topology/Cluster");
  4425. ForEach(*clusterIter)
  4426. {
  4427. IPropertyTree &node = clusterIter->query();
  4428. Owned<IConstWUClusterInfo> cluster = getTargetClusterInfo(environmentRoot, &node);
  4429. clusters.append(*cluster.getClear());
  4430. }
  4431. return clusters.ordinality();
  4432. }
  4433. const char *getTargetClusterComponentName(const char *clustname, const char *processType, StringBuffer &name)
  4434. {
  4435. if (!clustname)
  4436. return NULL;
  4437. Owned<IEnvironmentFactory> factory = getEnvironmentFactory();
  4438. Owned<IConstEnvironment> env = factory->openEnvironment();
  4439. if (!env)
  4440. return NULL;
  4441. Owned<IPropertyTree> root = &env->getPTree();
  4442. StringBuffer xpath;
  4443. xpath.appendf("Software/Topology/Cluster[@name=\"%s\"]", clustname);
  4444. Owned<IPropertyTree> cluster = root->getPropTree(xpath.str());
  4445. if (!cluster)
  4446. return NULL;
  4447. StringBuffer xpath1;
  4448. xpath1.appendf("%s/@process", processType);
  4449. name.append(cluster->queryProp(xpath1.str()));
  4450. return name.str();
  4451. }
  4452. unsigned getEnvironmentThorClusterNames(StringArray &thorNames, StringArray &groupNames, StringArray &targetNames, StringArray &queueNames)
  4453. {
  4454. Owned<IEnvironmentFactory> factory = getEnvironmentFactory();
  4455. Owned<IConstEnvironment> env = factory->openEnvironment();
  4456. if (!env)
  4457. return 0;
  4458. Owned<IPropertyTree> root = &env->getPTree();
  4459. Owned<IPropertyTreeIterator> allTargets = root->getElements("Software/Topology/Cluster");
  4460. ForEach(*allTargets)
  4461. {
  4462. IPropertyTree &target = allTargets->query();
  4463. const char *targetName = target.queryProp("@name");
  4464. if (targetName && *targetName)
  4465. {
  4466. Owned<IPropertyTreeIterator> thorClusters = target.getElements("ThorCluster");
  4467. ForEach(*thorClusters)
  4468. {
  4469. const char *thorName = thorClusters->query().queryProp("@process");
  4470. VStringBuffer query("Software/ThorCluster[@name=\"%s\"]",thorName);
  4471. IPropertyTree *thorCluster = root->queryPropTree(query.str());
  4472. if (thorCluster)
  4473. {
  4474. const char *groupName = thorCluster->queryProp("@nodeGroup");
  4475. if (!groupName||!*groupName)
  4476. groupName = thorName;
  4477. thorNames.append(thorName);
  4478. groupNames.append(groupName);
  4479. targetNames.append(targetName);
  4480. StringBuffer queueName(targetName);
  4481. queueNames.append(queueName.append(THOR_QUEUE_EXT));
  4482. }
  4483. }
  4484. }
  4485. }
  4486. return thorNames.ordinality();
  4487. }
  4488. unsigned getEnvironmentHThorClusterNames(StringArray &eclAgentNames, StringArray &groupNames, StringArray &targetNames)
  4489. {
  4490. Owned<IEnvironmentFactory> factory = getEnvironmentFactory();
  4491. Owned<IConstEnvironment> env = factory->openEnvironment();
  4492. if (!env)
  4493. return 0;
  4494. Owned<IPropertyTree> root = &env->getPTree();
  4495. Owned<IPropertyTreeIterator> allEclAgents = root->getElements("Software/EclAgentProcess");
  4496. ForEach(*allEclAgents)
  4497. {
  4498. IPropertyTree &eclAgent = allEclAgents->query();
  4499. const char *eclAgentName = eclAgent.queryProp("@name");
  4500. if (eclAgentName && *eclAgentName)
  4501. {
  4502. Owned<IPropertyTreeIterator> allTargets = root->getElements("Software/Topology/Cluster");
  4503. ForEach(*allTargets)
  4504. {
  4505. IPropertyTree &target = allTargets->query();
  4506. const char *targetName = target.queryProp("@name");
  4507. if (targetName && *targetName)
  4508. {
  4509. StringBuffer xpath;
  4510. xpath.appendf("EclAgentProcess[@process=\"%s\"]", eclAgentName);
  4511. if (target.hasProp(xpath) && !target.hasProp("ThorCluster"))
  4512. {
  4513. StringBuffer groupName("hthor__");
  4514. groupName.append(eclAgentName);
  4515. groupNames.append(groupName);
  4516. eclAgentNames.append(eclAgentName);
  4517. targetNames.append(targetName);
  4518. }
  4519. }
  4520. }
  4521. }
  4522. }
  4523. return eclAgentNames.ordinality();
  4524. }
  4525. IStringVal& CLocalWorkUnit::getScope(IStringVal &str) const
  4526. {
  4527. CriticalBlock block(crit);
  4528. if (p->hasProp("Debug/ForceScope"))
  4529. {
  4530. StringBuffer prefix(p->queryProp("Debug/ForceScope"));
  4531. str.set(prefix.toLowerCase().str());
  4532. }
  4533. else
  4534. {
  4535. Owned <IConstWUClusterInfo> ci = getTargetClusterInfo(p->queryProp("@clusterName"));
  4536. if (ci)
  4537. ci->getScope(str);
  4538. else
  4539. str.clear();
  4540. }
  4541. return str;
  4542. }
  4543. //Queries
  4544. void CLocalWorkUnit::setCodeVersion(unsigned codeVersion, const char * buildVersion, const char * eclVersion)
  4545. {
  4546. CriticalBlock block(crit);
  4547. p->setPropInt("@codeVersion", codeVersion);
  4548. p->setProp("@buildVersion", buildVersion);
  4549. p->setProp("@eclVersion", eclVersion);
  4550. }
  4551. unsigned CLocalWorkUnit::getCodeVersion() const
  4552. {
  4553. CriticalBlock block(crit);
  4554. return p->getPropInt("@codeVersion");
  4555. }
  4556. unsigned CLocalWorkUnit::getWuidVersion() const
  4557. {
  4558. CriticalBlock block(crit);
  4559. return p->getPropInt("@wuidVersion");
  4560. }
  4561. void CLocalWorkUnit::getBuildVersion(IStringVal & buildVersion, IStringVal & eclVersion) const
  4562. {
  4563. CriticalBlock block(crit);
  4564. buildVersion.set(p->queryProp("@buildVersion"));
  4565. eclVersion.set(p->queryProp("@eclVersion"));
  4566. }
  4567. void CLocalWorkUnit::setCloneable(bool value)
  4568. {
  4569. CriticalBlock block(crit);
  4570. p->setPropInt("@cloneable", value);
  4571. }
  4572. void CLocalWorkUnit::setIsClone(bool value)
  4573. {
  4574. CriticalBlock block(crit);
  4575. p->setPropInt("@isClone", value);
  4576. }
  4577. bool CLocalWorkUnit::getCloneable() const
  4578. {
  4579. CriticalBlock block(crit);
  4580. return p->getPropBool("@cloneable", false);
  4581. }
  4582. IUserDescriptor *CLocalWorkUnit::queryUserDescriptor() const
  4583. {
  4584. CriticalBlock block(crit);
  4585. if (!userDesc)
  4586. {
  4587. SCMStringBuffer token, user, password;
  4588. getSecurityToken(token);
  4589. extractToken(token.str(), queryWuid(), user, password);
  4590. userDesc.setown(createUserDescriptor());
  4591. userDesc->set(user.str(), password.str());
  4592. }
  4593. return userDesc;
  4594. }
  4595. bool CLocalWorkUnit::isProtected() const
  4596. {
  4597. CriticalBlock block(crit);
  4598. return p->getPropBool("@protected", false);
  4599. }
  4600. bool CLocalWorkUnit::isPausing() const
  4601. {
  4602. CriticalBlock block(crit);
  4603. if (WUActionPause == getAction())
  4604. {
  4605. switch (getState())
  4606. {
  4607. case WUStateRunning:
  4608. case WUStateAborting:
  4609. return true;
  4610. }
  4611. }
  4612. return false;
  4613. }
  4614. void CLocalWorkUnit::protect(bool protectMode)
  4615. {
  4616. CriticalBlock block(crit);
  4617. p->setPropBool("@protected", protectMode);
  4618. }
  4619. void CLocalWorkUnit::setResultLimit(unsigned value)
  4620. {
  4621. CriticalBlock block(crit);
  4622. p->setPropInt("resultLimit", value);
  4623. }
  4624. unsigned CLocalWorkUnit::getResultLimit() const
  4625. {
  4626. CriticalBlock block(crit);
  4627. return p->getPropInt("resultLimit");
  4628. }
  4629. IStringVal & CLocalWorkUnit::getSnapshot(IStringVal & str) const
  4630. {
  4631. CriticalBlock block(crit);
  4632. str.set(p->queryProp("SNAPSHOT"));
  4633. return str;
  4634. }
  4635. void CLocalWorkUnit::setSnapshot(const char * val)
  4636. {
  4637. CriticalBlock block(crit);
  4638. p->setProp("SNAPSHOT", val);
  4639. }
  4640. const static mapEnums warningSeverityMap[] =
  4641. {
  4642. { SeverityInformation, "info" },
  4643. { SeverityWarning, "warning" },
  4644. { SeverityError, "error" },
  4645. { SeverityAlert, "alert" },
  4646. { SeverityIgnore, "ignore" },
  4647. { SeverityFatal, "fatal" },
  4648. { SeverityUnknown, NULL }
  4649. };
  4650. ErrorSeverity CLocalWorkUnit::getWarningSeverity(unsigned code, ErrorSeverity defaultSeverity) const
  4651. {
  4652. StringBuffer xpath;
  4653. xpath.append("OnWarnings/OnWarning[@code='").append(code).append("']");
  4654. CriticalBlock block(crit);
  4655. IPropertyTree * mapping = p->queryPropTree(xpath);
  4656. if (mapping)
  4657. return (ErrorSeverity) getEnum(mapping, "@severity", warningSeverityMap);
  4658. return defaultSeverity;
  4659. }
  4660. void CLocalWorkUnit::setWarningSeverity(unsigned code, ErrorSeverity severity)
  4661. {
  4662. StringBuffer xpath;
  4663. xpath.append("OnWarnings/OnWarning[@code='").append(code).append("']");
  4664. CriticalBlock block(crit);
  4665. IPropertyTree * mapping = p->queryPropTree(xpath);
  4666. if (!mapping)
  4667. {
  4668. IPropertyTree * onWarnings = ensurePTree(p, "OnWarnings");
  4669. mapping = onWarnings->addPropTree("OnWarning", createPTree());
  4670. mapping->setPropInt("@code", code);
  4671. }
  4672. setEnum(mapping, "@severity", severity, warningSeverityMap);
  4673. }
  4674. static int comparePropTrees(IInterface * const *ll, IInterface * const *rr)
  4675. {
  4676. IPropertyTree *l = (IPropertyTree *) *ll;
  4677. IPropertyTree *r = (IPropertyTree *) *rr;
  4678. return stricmp(l->queryName(), r->queryName());
  4679. };
  4680. unsigned CLocalWorkUnit::calculateHash(unsigned crc)
  4681. {
  4682. // Any other values in the WU that could affect generated code should be crc'ed here
  4683. IPropertyTree *tree = p->queryBranch("Debug");
  4684. if (tree)
  4685. {
  4686. Owned<IPropertyTreeIterator> sub = tree->getElements("*");
  4687. ICopyArrayOf<IPropertyTree> subs;
  4688. for(sub->first(); sub->isValid(); sub->next())
  4689. subs.append(sub->query());
  4690. subs.sort(comparePropTrees);
  4691. ForEachItemIn(idx, subs)
  4692. {
  4693. const char *name = subs.item(idx).queryName();
  4694. const char *val = subs.item(idx).queryProp(NULL);
  4695. crc = crc32(name, (size32_t)strlen(name), crc);
  4696. if (val)
  4697. crc = crc32(val, (size32_t)strlen(val), crc);
  4698. }
  4699. }
  4700. Owned<IConstWUPluginIterator> plugins = &getPlugins();
  4701. for (plugins->first();plugins->isValid();plugins->next())
  4702. {
  4703. IConstWUPlugin &thisplugin = plugins->query();
  4704. SCMStringBuffer version;
  4705. thisplugin.getPluginVersion(version);
  4706. crc = crc32(version.str(), version.length(), crc);
  4707. }
  4708. return crc;
  4709. }
  4710. static void updateProp(IPropertyTree * to, const IPropertyTree * from, const char * xpath)
  4711. {
  4712. if (!to->hasProp(xpath) && from->hasProp(xpath))
  4713. to->setProp(xpath, from->queryProp(xpath));
  4714. }
  4715. static void setProp(IPropertyTree * to, const IPropertyTree * from, const char * xpath)
  4716. {
  4717. if (from->hasProp(xpath))
  4718. to->setProp(xpath, from->queryProp(xpath));
  4719. }
  4720. static void copyTree(IPropertyTree * to, const IPropertyTree * from, const char * xpath)
  4721. {
  4722. IPropertyTree * match = from->getBranch(xpath);
  4723. if (match)
  4724. to->setPropTree(xpath, match);
  4725. }
  4726. IPropertyTree *CLocalWorkUnit::queryPTree() const
  4727. {
  4728. return p;
  4729. }
  4730. void CLocalWorkUnit::copyWorkUnit(IConstWorkUnit *cached, bool all)
  4731. {
  4732. CLocalWorkUnit *from = QUERYINTERFACE(cached, CLocalWorkUnit);
  4733. if (!from)
  4734. {
  4735. CLockedWorkUnit *fl = QUERYINTERFACE(cached, CLockedWorkUnit);
  4736. if (!fl)
  4737. throw MakeStringException(WUERR_InternalUnknownImplementation, "Cached workunit not created using workunit dll");
  4738. from = fl->c;
  4739. }
  4740. // Need to copy the query, the results, and the graphs from the cached query.
  4741. // The cache is made before the query is executed so there is no need to clear them.
  4742. if (!cached->getCloneable())
  4743. throw MakeStringException(WUERR_CannotCloneWorkunit, "Source work unit not marked as clonable");
  4744. const IPropertyTree * fromP = from->p;
  4745. IPropertyTree *pt;
  4746. CriticalBlock block(crit);
  4747. clearCached(false);
  4748. query.clear();
  4749. updateProp(p, fromP, "@jobName");
  4750. copyTree(p, fromP, "Query");
  4751. pt = fromP->getBranch("Application/LibraryModule");
  4752. if (pt)
  4753. {
  4754. ensurePTree(p, "Application");
  4755. p->setPropTree("Application/LibraryModule", pt);
  4756. }
  4757. pt = fromP->queryBranch("Debug");
  4758. if (pt)
  4759. {
  4760. IPropertyTree *curDebug = p->queryPropTree("Debug");
  4761. if (curDebug)
  4762. {
  4763. Owned<IPropertyTreeIterator> elems = pt->getElements("*");
  4764. ForEach(*elems)
  4765. {
  4766. IPropertyTree *elem = &elems->query();
  4767. if (!curDebug->hasProp(elem->queryName()))
  4768. curDebug->setPropTree(elem->queryName(),LINK(elem));
  4769. }
  4770. }
  4771. else
  4772. p->setPropTree("Debug", LINK(pt));
  4773. }
  4774. copyTree(p, fromP, "OnWarnings");
  4775. copyTree(p, fromP, "Plugins");
  4776. copyTree(p, fromP, "Libraries");
  4777. copyTree(p, fromP, "Results");
  4778. copyTree(p, fromP, "Graphs");
  4779. copyTree(p, fromP, "Workflow");
  4780. copyTree(p, fromP, "WebServicesInfo");
  4781. if (all)
  4782. {
  4783. // 'all' mode is used when setting up a dali WU from the embedded wu in a workunit dll
  4784. // Merge timing info from both branches
  4785. pt = fromP->getBranch("Timings");
  4786. if (pt)
  4787. {
  4788. IPropertyTree *tgtTimings = ensurePTree(p, "Timings");
  4789. mergePTree(tgtTimings, pt);
  4790. pt->Release();
  4791. }
  4792. pt = fromP->getBranch("Statistics");
  4793. if (pt)
  4794. {
  4795. IPropertyTree *tgtStatistics = ensurePTree(p, "Statistics");
  4796. mergePTree(tgtStatistics, pt);
  4797. pt->Release();
  4798. }
  4799. }
  4800. updateProp(p, fromP, "@clusterName");
  4801. updateProp(p, fromP, "allowedclusters");
  4802. updateProp(p, fromP, "@submitID");
  4803. updateProp(p, fromP, "SNAPSHOT");
  4804. //MORE: This is very adhoc. All options that should be cloned should really be in a common branch
  4805. if (all)
  4806. {
  4807. setProp(p, fromP, "PriorityFlag");
  4808. setProp(p, fromP, "@priorityClass");
  4809. setProp(p, fromP, "@protected");
  4810. setProp(p, fromP, "@clusterName");
  4811. updateProp(p, fromP, "@scope");
  4812. }
  4813. //Variables may have been set up as parameters to the query - so need to preserve any values that were supplied.
  4814. pt = fromP->getBranch("Variables");
  4815. if (pt)
  4816. {
  4817. IPropertyTree *ptTgtVariables = ensurePTree(p, "Variables");
  4818. Owned<IPropertyTreeIterator> ptiVariable = pt->getElements("Variable");
  4819. for (ptiVariable->first(); ptiVariable->isValid(); ptiVariable->next())
  4820. {
  4821. IPropertyTree *ptSrcVariable = &ptiVariable->query();
  4822. const char *name = ptSrcVariable->queryProp("@name");
  4823. assertex(name);
  4824. StringBuffer xpath;
  4825. xpath.append("Variable[@name='").append(name).append("']");
  4826. IPropertyTree *ptTgtVariable = ptTgtVariables->queryPropTree(xpath.str());
  4827. IPropertyTree *merged = createPTreeFromIPT(ptSrcVariable); // clone entire source info...
  4828. merged->removeProp("Value"); // except value and status
  4829. merged->setProp("@status", "undefined");
  4830. if (!merged->getPropBool("@isScalar"))
  4831. merged->removeProp("totalRowCount");
  4832. merged->removeProp("rowCount");
  4833. // If there are any other fields that get set ONLY by eclagent, strip them out here...
  4834. if (ptTgtVariable)
  4835. {
  4836. // copy status and Value from what is already set in target
  4837. merged->setProp("@status", ptTgtVariable->queryProp("@status"));
  4838. MemoryBuffer value;
  4839. if (ptTgtVariable->getPropBin("Value", value))
  4840. merged->setPropBin("Value", value.length(), value.toByteArray());
  4841. ptTgtVariable->removeProp(xpath.str());
  4842. // If there are any other fields in a variable that get set by ws_ecl before submitting, copy them across here...
  4843. }
  4844. ptTgtVariables->addPropTree("Variable", merged);
  4845. }
  4846. pt->Release();
  4847. }
  4848. p->setProp("@codeVersion", fromP->queryProp("@codeVersion"));
  4849. p->setProp("@buildVersion", fromP->queryProp("@buildVersion"));
  4850. p->setProp("@eclVersion", fromP->queryProp("@eclVersion"));
  4851. p->setProp("@hash", fromP->queryProp("@hash"));
  4852. p->setPropBool("@cloneable", true);
  4853. p->setPropBool("@isClone", true);
  4854. resetWorkflow(); // the source Workflow section may have had some parts already executed...
  4855. // resetResults(); // probably should be resetting the results as well... rather than waiting for the rerun to overwrite them
  4856. }
  4857. bool CLocalWorkUnit::hasDebugValue(const char *propname) const
  4858. {
  4859. StringBuffer lower;
  4860. lower.append(propname).toLowerCase();
  4861. CriticalBlock block(crit);
  4862. StringBuffer prop("Debug/");
  4863. return p->hasProp(prop.append(lower));
  4864. }
  4865. IStringVal& CLocalWorkUnit::getDebugValue(const char *propname, IStringVal &str) const
  4866. {
  4867. StringBuffer lower;
  4868. lower.append(propname).toLowerCase();
  4869. CriticalBlock block(crit);
  4870. StringBuffer prop("Debug/");
  4871. str.set(p->queryProp(prop.append(lower).str()));
  4872. return str;
  4873. }
  4874. IStringIterator& CLocalWorkUnit::getDebugValues() const
  4875. {
  4876. return getDebugValues(NULL);
  4877. }
  4878. IStringIterator& CLocalWorkUnit::getDebugValues(const char *prop) const
  4879. {
  4880. CriticalBlock block(crit);
  4881. StringBuffer path("Debug/");
  4882. if (prop)
  4883. {
  4884. StringBuffer lower;
  4885. lower.append(prop).toLowerCase();
  4886. path.append(lower);
  4887. }
  4888. else
  4889. path.append("*");
  4890. return *new CStringPTreeTagIterator(p->getElements(path.str()));
  4891. }
  4892. int CLocalWorkUnit::getDebugValueInt(const char *propname, int defVal) const
  4893. {
  4894. StringBuffer lower;
  4895. lower.append(propname).toLowerCase();
  4896. CriticalBlock block(crit);
  4897. StringBuffer prop("Debug/");
  4898. prop.append(lower);
  4899. return p->getPropInt(prop.str(), defVal);
  4900. }
  4901. __int64 CLocalWorkUnit::getDebugValueInt64(const char *propname, __int64 defVal) const
  4902. {
  4903. StringBuffer lower;
  4904. lower.append(propname).toLowerCase();
  4905. CriticalBlock block(crit);
  4906. StringBuffer prop("Debug/");
  4907. prop.append(lower);
  4908. return p->getPropInt64(prop.str(), defVal);
  4909. }
  4910. bool CLocalWorkUnit::getDebugValueBool(const char * propname, bool defVal) const
  4911. {
  4912. StringBuffer lower;
  4913. lower.append(propname).toLowerCase();
  4914. CriticalBlock block(crit);
  4915. StringBuffer prop("Debug/");
  4916. prop.append(lower);
  4917. return p->getPropBool(prop.str(), defVal);
  4918. }
  4919. IStringIterator *CLocalWorkUnit::getLogs(const char *type, const char *instance) const
  4920. {
  4921. VStringBuffer xpath("Process/%s/", type);
  4922. if (instance)
  4923. xpath.append(instance);
  4924. else
  4925. xpath.append("*");
  4926. CriticalBlock block(crit);
  4927. if (p->getPropInt("@wuidVersion") < 1) // legacy wuid
  4928. {
  4929. // NB: instance unused
  4930. if (streq("EclAgent", type))
  4931. return new CStringPTreeIterator(p->getElements("Debug/eclagentlog"));
  4932. else if (streq("Thor", type))
  4933. return new CStringPTreeIterator(p->getElements("Debug/thorlog*"));
  4934. VStringBuffer xpath("Debug/%s", type);
  4935. return new CStringPTreeIterator(p->getElements(xpath.str()));
  4936. }
  4937. else
  4938. return new CStringPTreeAttrIterator(p->getElements(xpath.str()), "@log");
  4939. }
  4940. IPropertyTreeIterator* CLocalWorkUnit::getProcesses(const char *type, const char *instance) const
  4941. {
  4942. VStringBuffer xpath("Process/%s/", type);
  4943. if (instance)
  4944. xpath.append(instance);
  4945. else
  4946. xpath.append("*");
  4947. CriticalBlock block(crit);
  4948. return p->getElements(xpath.str());
  4949. }
  4950. IStringIterator *CLocalWorkUnit::getProcesses(const char *type) const
  4951. {
  4952. VStringBuffer xpath("Process/%s/*", type);
  4953. CriticalBlock block(crit);
  4954. return new CStringPTreeTagIterator(p->getElements(xpath.str()));
  4955. }
  4956. void CLocalWorkUnit::addProcess(const char *type, const char *instance, unsigned pid, const char *log)
  4957. {
  4958. VStringBuffer processType("Process/%s", type);
  4959. VStringBuffer xpath("%s/%s", processType.str(), instance);
  4960. if (log)
  4961. xpath.appendf("[@log=\"%s\"]", log);
  4962. CriticalBlock block(crit);
  4963. if (!p->hasProp(xpath))
  4964. {
  4965. IPropertyTree *node = ensurePTree(p, processType.str());
  4966. node = node->addPropTree(instance, createPTree());
  4967. node->setProp("@log", log);
  4968. node->setPropInt("@pid", pid);
  4969. }
  4970. }
  4971. void CLocalWorkUnit::setDebugValue(const char *propname, const char *value, bool overwrite)
  4972. {
  4973. StringBuffer lower;
  4974. lower.append(propname).toLowerCase();
  4975. CriticalBlock block(crit);
  4976. StringBuffer prop("Debug/");
  4977. prop.append(lower);
  4978. if (overwrite || !p->hasProp(prop.str()))
  4979. {
  4980. // MORE - not sure this line should be needed....
  4981. p->setProp("Debug", "");
  4982. p->setProp(prop.str(), value);
  4983. }
  4984. }
  4985. void CLocalWorkUnit::setDebugValueInt(const char *propname, int value, bool overwrite)
  4986. {
  4987. StringBuffer lower;
  4988. lower.append(propname).toLowerCase();
  4989. CriticalBlock block(crit);
  4990. StringBuffer prop("Debug/");
  4991. prop.append(lower);
  4992. if (overwrite || !p->hasProp(prop.str()))
  4993. {
  4994. // MORE - not sure this line should be needed....
  4995. p->setProp("Debug", "");
  4996. p->setPropInt(prop.str(), value);
  4997. }
  4998. }
  4999. void CLocalWorkUnit::setTracingValue(const char *propname, const char *value)
  5000. {
  5001. CriticalBlock block(crit);
  5002. // MORE - not sure this line should be needed....
  5003. p->setProp("Tracing", "");
  5004. StringBuffer prop("Tracing/");
  5005. p->setProp(prop.append(propname).str(), value);
  5006. }
  5007. void CLocalWorkUnit::setTracingValueInt(const char *propname, int value)
  5008. {
  5009. CriticalBlock block(crit);
  5010. StringBuffer prop("Tracing/");
  5011. p->setPropInt(prop.append(propname).str(), value);
  5012. }
  5013. IConstWUQuery* CLocalWorkUnit::getQuery() const
  5014. {
  5015. // For this to be legally called, we must have the read-able interface. So we are already locked for (at least) read.
  5016. CriticalBlock block(crit);
  5017. if (!query)
  5018. {
  5019. IPropertyTree *s = p->getPropTree("Query");
  5020. if (s)
  5021. query.setown(new CLocalWUQuery(s)); // NB takes ownership of 's'
  5022. }
  5023. return query.getLink();
  5024. }
  5025. IWUQuery* CLocalWorkUnit::updateQuery()
  5026. {
  5027. // For this to be legally called, we must have the write-able interface. So we are already locked for write.
  5028. CriticalBlock block(crit);
  5029. if (!query)
  5030. {
  5031. IPropertyTree *s = p->queryPropTree("Query");
  5032. if (!s)
  5033. s = p->addPropTree("Query", createPTreeFromXMLString("<Query fetchEntire='1'/>"));
  5034. s->Link();
  5035. query.setown(new CLocalWUQuery(s));
  5036. }
  5037. return query.getLink();
  5038. }
  5039. void CLocalWorkUnit::loadPlugins() const
  5040. {
  5041. CriticalBlock block(crit);
  5042. if (!pluginsCached)
  5043. {
  5044. assertex(plugins.length() == 0);
  5045. Owned<IPropertyTreeIterator> r = p->getElements("Plugins/Plugin");
  5046. for (r->first(); r->isValid(); r->next())
  5047. {
  5048. IPropertyTree *rp = &r->query();
  5049. rp->Link();
  5050. plugins.append(*new CLocalWUPlugin(rp));
  5051. }
  5052. pluginsCached = true;
  5053. }
  5054. }
  5055. IConstWUPluginIterator& CLocalWorkUnit::getPlugins() const
  5056. {
  5057. CriticalBlock block(crit);
  5058. loadPlugins();
  5059. return *new CArrayIteratorOf<IConstWUPlugin,IConstWUPluginIterator> (plugins, 0, (IConstWorkUnit *) this);
  5060. }
  5061. void CLocalWorkUnit::loadLibraries() const
  5062. {
  5063. CriticalBlock block(crit);
  5064. if (!librariesCached)
  5065. {
  5066. assertex(libraries.length() == 0);
  5067. Owned<IPropertyTreeIterator> r = p->getElements("Libraries/Library");
  5068. ForEach(*r)
  5069. {
  5070. IPropertyTree *rp = &r->query();
  5071. rp->Link();
  5072. libraries.append(*new CLocalWULibrary(rp));
  5073. }
  5074. librariesCached = true;
  5075. }
  5076. }
  5077. IConstWULibraryIterator& CLocalWorkUnit::getLibraries() const
  5078. {
  5079. CriticalBlock block(crit);
  5080. loadLibraries();
  5081. return *new CArrayIteratorOf<IConstWULibrary,IConstWULibraryIterator> (libraries, 0, (IConstWorkUnit *) this);
  5082. }
  5083. IConstWULibrary * CLocalWorkUnit::getLibraryByName(const char * search) const
  5084. {
  5085. CriticalBlock block(crit);
  5086. loadLibraries();
  5087. ForEachItemIn(idx, libraries)
  5088. {
  5089. SCMStringBuffer name;
  5090. IConstWULibrary &cur = libraries.item(idx);
  5091. cur.getName(name);
  5092. if (stricmp(name.str(), search)==0)
  5093. return &OLINK(cur);
  5094. }
  5095. return NULL;
  5096. }
  5097. StringBuffer &formatGraphTimerLabel(StringBuffer &str, const char *graphName, unsigned subGraphNum, unsigned __int64 subId)
  5098. {
  5099. str.append("Graph ").append(graphName);
  5100. if (subGraphNum) str.append(" - ").append(subGraphNum).append(" (").append(subId).append(")");
  5101. else if (subId) str.append(" - id(").append(subId).append(")");
  5102. return str;
  5103. }
  5104. StringBuffer &formatGraphTimerScope(StringBuffer &str, const char *graphName, unsigned subGraphNum, unsigned __int64 subId)
  5105. {
  5106. str.append(graphName);
  5107. if (subId) str.append(":sg").append(subId);
  5108. return str;
  5109. }
  5110. bool parseGraphTimerLabel(const char *label, StringAttr &graphName, unsigned & graphNum, unsigned &subGraphNum, unsigned &subId)
  5111. {
  5112. // expects format: "Graph <graphname>[ - <subgraphnum> (<subgraphid>)]"
  5113. unsigned len = (size32_t)strlen(label);
  5114. if (len < 6 || (0 != memcmp(label, "Graph ", 6)))
  5115. return false;
  5116. graphNum = 0;
  5117. subGraphNum = 0;
  5118. subId = 0;
  5119. const char *finger = label+6;
  5120. const char *finger2 = strchr(finger, '-');
  5121. if (NULL == finger2) // just graphName
  5122. graphName.set(finger);
  5123. else
  5124. {
  5125. graphName.set(finger, (size32_t)((finger2-1)-finger));
  5126. finger = finger2+2; // skip '-' and space
  5127. finger2 = strchr(finger, ' ');
  5128. if (finger2)
  5129. {
  5130. subGraphNum = atoi_l(finger, (size32_t)(finger2-finger));
  5131. finger = finger2+2; // skip space and '('
  5132. finger2 = strchr(finger, ')');
  5133. if (finger2)
  5134. subId = atoi_l(finger, (size32_t)(finger2-finger));
  5135. }
  5136. else if (((len-(finger-label))>3) && 0 == memcmp(finger, "id(", 3)) // subgraph id only, new format.
  5137. {
  5138. finger += 3;
  5139. finger2 = strchr(finger, ')');
  5140. if (finger2)
  5141. subId = atoi_l(finger, (size32_t)(finger2-finger));
  5142. }
  5143. }
  5144. if (graphName && !memicmp(graphName, "graph", 5))
  5145. graphNum = atoi(graphName + 5);
  5146. return true;
  5147. }
  5148. bool parseGraphScope(const char *scope, StringAttr &graphName, unsigned & graphNum, unsigned &subGraphId)
  5149. {
  5150. if (!MATCHES_CONST_PREFIX(scope, GraphScopePrefix))
  5151. return false;
  5152. graphNum = atoi(scope + CONST_STRLEN(GraphScopePrefix));
  5153. const char * colon = strchr(scope, ':');
  5154. if (!colon)
  5155. {
  5156. graphName.set(scope);
  5157. subGraphId = 0;
  5158. return true;
  5159. }
  5160. const char * subgraph = colon+1;
  5161. graphName.set(scope, (size32_t)(colon - scope));
  5162. if (MATCHES_CONST_PREFIX(subgraph, SubGraphScopePrefix))
  5163. subGraphId = atoi(subgraph+CONST_STRLEN(SubGraphScopePrefix));
  5164. return true;
  5165. }
  5166. class WorkUnitStatisticsIterator : public CArrayIteratorOf<IConstWUStatistic,IConstWUStatisticIterator>
  5167. {
  5168. typedef CArrayIteratorOf<IConstWUStatistic,IConstWUStatisticIterator> PARENT;
  5169. public:
  5170. WorkUnitStatisticsIterator(const IArray &a, aindex_t start, IInterface *owner, const IStatisticsFilter * _filter)
  5171. : PARENT(a,start, owner), filter(_filter)
  5172. {
  5173. }
  5174. virtual bool first()
  5175. {
  5176. if (!PARENT::first())
  5177. return false;
  5178. if (matchesFilter())
  5179. return true;
  5180. return next();
  5181. }
  5182. virtual bool next()
  5183. {
  5184. loop
  5185. {
  5186. if (!PARENT::next())
  5187. return false;
  5188. if (matchesFilter())
  5189. return true;
  5190. }
  5191. }
  5192. protected:
  5193. bool matchesFilter()
  5194. {
  5195. if (!filter)
  5196. return true;
  5197. return query().matches(filter);
  5198. }
  5199. protected:
  5200. Linked<const IStatisticsFilter> filter;
  5201. };
  5202. void CLocalWorkUnit::setStatistic(StatisticCreatorType creatorType, const char * creator, StatisticScopeType scopeType, const char * scope, StatisticKind kind, const char * optDescription, unsigned __int64 value, unsigned __int64 count, unsigned __int64 maxValue, StatsMergeAction mergeAction)
  5203. {
  5204. if (!scope || !*scope) scope = GLOBAL_SCOPE;
  5205. const char * kindName = queryStatisticName(kind);
  5206. StatisticMeasure measure = queryMeasure(kind);
  5207. //creator. scope and name must all be present, and must not contain semi colons.
  5208. assertex(creator && scope);
  5209. CriticalBlock block(crit);
  5210. IPropertyTree * stats = p->queryPropTree("Statistics");
  5211. if (!stats)
  5212. stats = p->addPropTree("Statistics", createPTree("Statistics"));
  5213. IPropertyTree * statTree = NULL;
  5214. if (mergeAction != StatsMergeAppend)
  5215. {
  5216. StringBuffer xpath;
  5217. xpath.append("Statistic[@creator='").append(creator).append("'][@scope='").append(scope).append("'][@kind='").append(kindName).append("']");
  5218. statTree = stats->queryPropTree(xpath.str());
  5219. }
  5220. if (!statTree)
  5221. {
  5222. statTree = stats->addPropTree("Statistic", createPTree("Statistic"));
  5223. statTree->setProp("@creator", creator);
  5224. statTree->setProp("@scope", scope);
  5225. statTree->setProp("@kind", kindName);
  5226. //These items are primarily here to facilitate filtering.
  5227. statTree->setProp("@unit", queryMeasureName(measure));
  5228. statTree->setProp("@c", queryCreatorTypeName(creatorType));
  5229. statTree->setProp("@s", queryScopeTypeName(scopeType));
  5230. statTree->setPropInt64("@ts", getTimeStampNowValue());
  5231. if (optDescription)
  5232. statTree->setProp("@desc", optDescription);
  5233. if (statistics.cached)
  5234. statistics.append(LINK(statTree));
  5235. mergeAction = StatsMergeAppend;
  5236. }
  5237. if (mergeAction != StatsMergeAppend)
  5238. {
  5239. unsigned __int64 oldValue = statTree->getPropInt64("@value", 0);
  5240. unsigned __int64 oldCount = statTree->getPropInt64("@count", 0);
  5241. unsigned __int64 oldMax = statTree->getPropInt64("@max", 0);
  5242. if (oldMax < oldValue)
  5243. oldMax = oldValue;
  5244. statTree->setPropInt64("@value", mergeStatisticValue(oldValue, value, mergeAction));
  5245. statTree->setPropInt64("@count", count + oldCount);
  5246. if (maxValue > oldMax)
  5247. statTree->setPropInt64("@max", maxValue);
  5248. }
  5249. else
  5250. {
  5251. statTree->setPropInt64("@value", value);
  5252. statTree->setPropInt64("@count", count);
  5253. if (maxValue)
  5254. statTree->setPropInt64("@max", maxValue);
  5255. else
  5256. statTree->removeProp("@max");
  5257. }
  5258. }
  5259. void CLocalWorkUnit::_loadStatistics() const
  5260. {
  5261. statistics.load(p,"Statistics/*");
  5262. }
  5263. IConstWUStatisticIterator& CLocalWorkUnit::getStatistics(const IStatisticsFilter * filter) const
  5264. {
  5265. CriticalBlock block(crit);
  5266. //This should be deleted in version 6.0 when support for 4.x is no longer required
  5267. legacyTimings.load(p,"Timings/*");
  5268. if (legacyTimings.ordinality())
  5269. return *new WorkUnitStatisticsIterator(legacyTimings, 0, (IConstWorkUnit *) this, filter);
  5270. statistics.load(p,"Statistics/*");
  5271. Owned<IConstWUStatisticIterator> localStats = new WorkUnitStatisticsIterator(statistics, 0, (IConstWorkUnit *) this, filter);
  5272. const char * wuid = p->queryName();
  5273. Owned<IConstWUStatisticIterator> graphStats = new CConstGraphProgressStatisticsIterator(wuid, filter);
  5274. return * new CCompoundIteratorOf<IConstWUStatisticIterator, IConstWUStatistic>(localStats, graphStats);
  5275. }
  5276. IConstWUStatistic * CLocalWorkUnit::getStatistic(const char * creator, const char * scope, StatisticKind kind) const
  5277. {
  5278. //MORE: Optimize this....
  5279. StatisticsFilter filter;
  5280. filter.setCreator(creator);
  5281. filter.setScope(scope);
  5282. filter.setKind(kind);
  5283. Owned<IConstWUStatisticIterator> stats = &getStatistics(&filter);
  5284. if (stats->first())
  5285. return LINK(&stats->query());
  5286. return NULL;
  5287. }
  5288. IWUPlugin* CLocalWorkUnit::updatePluginByName(const char *qname)
  5289. {
  5290. CriticalBlock block(crit);
  5291. IConstWUPlugin *existing = getPluginByName(qname);
  5292. if (existing)
  5293. return (IWUPlugin *) existing;
  5294. if (!plugins.length())
  5295. p->addPropTree("Plugins", createPTree("Plugins"));
  5296. IPropertyTree *pl = p->queryPropTree("Plugins");
  5297. IPropertyTree *s = pl->addPropTree("Plugin", createPTree("Plugin"));
  5298. s->Link();
  5299. IWUPlugin* q = new CLocalWUPlugin(s);
  5300. q->Link();
  5301. plugins.append(*q);
  5302. q->setPluginName(qname);
  5303. return q;
  5304. }
  5305. IConstWUPlugin* CLocalWorkUnit::getPluginByName(const char *qname) const
  5306. {
  5307. CriticalBlock block(crit);
  5308. loadPlugins();
  5309. ForEachItemIn(idx, plugins)
  5310. {
  5311. SCMStringBuffer name;
  5312. IConstWUPlugin &cur = plugins.item(idx);
  5313. cur.getPluginName(name);
  5314. if (stricmp(name.str(), qname)==0)
  5315. {
  5316. cur.Link();
  5317. return &cur;
  5318. }
  5319. }
  5320. return NULL;
  5321. }
  5322. IWULibrary* CLocalWorkUnit::updateLibraryByName(const char *qname)
  5323. {
  5324. CriticalBlock block(crit);
  5325. IConstWULibrary *existing = getLibraryByName(qname);
  5326. if (existing)
  5327. return (IWULibrary *) existing;
  5328. if (!libraries.length())
  5329. p->addPropTree("Libraries", createPTree("Libraries"));
  5330. IPropertyTree *pl = p->queryPropTree("Libraries");
  5331. IPropertyTree *s = pl->addPropTree("Library", createPTree("Library"));
  5332. s->Link();
  5333. IWULibrary* q = new CLocalWULibrary(s);
  5334. q->Link();
  5335. libraries.append(*q);
  5336. q->setName(qname);
  5337. return q;
  5338. }
  5339. void CLocalWorkUnit::unsubscribe()
  5340. {
  5341. // Only overriding versions need to do anything
  5342. }
  5343. void CLocalWorkUnit::_loadExceptions() const
  5344. {
  5345. assertex(exceptions.length() == 0);
  5346. Owned<IPropertyTreeIterator> r = p->getElements("Exceptions/Exception");
  5347. for (r->first(); r->isValid(); r->next())
  5348. {
  5349. IPropertyTree *rp = &r->query();
  5350. rp->Link();
  5351. exceptions.append(*new CLocalWUException(rp));
  5352. }
  5353. }
  5354. void CLocalWorkUnit::loadExceptions() const
  5355. {
  5356. CriticalBlock block(crit);
  5357. if (!exceptionsCached)
  5358. {
  5359. _loadExceptions();
  5360. exceptionsCached = true;
  5361. }
  5362. }
  5363. IConstWUExceptionIterator& CLocalWorkUnit::getExceptions() const
  5364. {
  5365. CriticalBlock block(crit);
  5366. loadExceptions();
  5367. return *new CArrayIteratorOf<IConstWUException,IConstWUExceptionIterator> (exceptions, 0, (IConstWorkUnit *) this);
  5368. }
  5369. unsigned CLocalWorkUnit::getExceptionCount() const
  5370. {
  5371. CriticalBlock block(crit);
  5372. loadExceptions();
  5373. return exceptions.length();
  5374. }
  5375. void CLocalWorkUnit::clearExceptions()
  5376. {
  5377. CriticalBlock block(crit);
  5378. // For this to be legally called, we must have the write-able interface. So we are already locked for write.
  5379. exceptions.kill();
  5380. exceptionsCached = true;
  5381. p->removeProp("Exceptions");
  5382. }
  5383. IWUException* CLocalWorkUnit::createException()
  5384. {
  5385. CriticalBlock block(crit);
  5386. // For this to be legally called, we must have the write-able interface. So we are already locked for write.
  5387. loadExceptions();
  5388. if (!exceptions.length())
  5389. p->addPropTree("Exceptions", createPTree("Exceptions"));
  5390. IPropertyTree *r = p->queryPropTree("Exceptions");
  5391. IPropertyTree *s = r->addPropTree("Exception", createPTree("Exception"));
  5392. s->setPropInt("@sequence", exceptions.ordinality());
  5393. IWUException* q = new CLocalWUException(LINK(s));
  5394. exceptions.append(*LINK(q));
  5395. Owned<IJlibDateTime> now = createDateTimeNow();
  5396. SCMStringBuffer temp;
  5397. now->getString(temp);
  5398. q->setTimeStamp(temp.str());
  5399. return q;
  5400. }
  5401. IConstWUWebServicesInfo* CLocalWorkUnit::getWebServicesInfo() const
  5402. {
  5403. // For this to be legally called, we must have the read-able interface. So we are already locked for (at least) read.
  5404. CriticalBlock block(crit);
  5405. if (!webServicesInfoCached)
  5406. {
  5407. assertex(!webServicesInfo);
  5408. IPropertyTree *s = p->getPropTree("WebServicesInfo");
  5409. if (s)
  5410. webServicesInfo.setown(new CLocalWUWebServicesInfo(s)); // NB takes ownership of 's'
  5411. webServicesInfoCached = true;
  5412. }
  5413. return webServicesInfo.getLink();
  5414. }
  5415. IWUWebServicesInfo* CLocalWorkUnit::updateWebServicesInfo(bool create)
  5416. {
  5417. // For this to be legally called, we must have the write-able interface. So we are already locked for write.
  5418. CriticalBlock block(crit);
  5419. if (!webServicesInfoCached)
  5420. {
  5421. IPropertyTree *s = p->queryPropTree("WebServicesInfo");
  5422. if (!s)
  5423. {
  5424. if (create)
  5425. s = p->addPropTree("WebServicesInfo", createPTreeFromXMLString("<WebServicesInfo />"));
  5426. else
  5427. return NULL;
  5428. }
  5429. s->Link();
  5430. webServicesInfo.setown(new CLocalWUWebServicesInfo(s));
  5431. webServicesInfoCached = true;
  5432. }
  5433. return webServicesInfo.getLink();
  5434. }
  5435. IConstWURoxieQueryInfo* CLocalWorkUnit::getRoxieQueryInfo() const
  5436. {
  5437. // For this to be legally called, we must have the read-able interface. So we are already locked for (at least) read.
  5438. CriticalBlock block(crit);
  5439. if (!roxieQueryInfoCached)
  5440. {
  5441. assertex(!roxieQueryInfo);
  5442. IPropertyTree *s = p->getPropTree("RoxieQueryInfo");
  5443. if (s)
  5444. roxieQueryInfo.setown(new CLocalWURoxieQueryInfo(s)); // NB takes ownership of 's'
  5445. roxieQueryInfoCached = true;
  5446. }
  5447. return roxieQueryInfo.getLink();
  5448. }
  5449. IWURoxieQueryInfo* CLocalWorkUnit::updateRoxieQueryInfo(const char *wuid, const char *roxieClusterName)
  5450. {
  5451. // For this to be legally called, we must have the write-able interface. So we are already locked for write.
  5452. CriticalBlock block(crit);
  5453. if (!roxieQueryInfoCached)
  5454. {
  5455. IPropertyTree *s = p->queryPropTree("RoxieQueryInfo");
  5456. if (!s)
  5457. s = p->addPropTree("RoxieQueryInfo", createPTreeFromXMLString("<RoxieQueryInfo />"));
  5458. if (wuid && *wuid)
  5459. s->addProp("@wuid", wuid);
  5460. if (roxieClusterName && *roxieClusterName)
  5461. s->addProp("@roxieClusterName", roxieClusterName);
  5462. s->Link();
  5463. roxieQueryInfo.setown(new CLocalWURoxieQueryInfo(s));
  5464. roxieQueryInfoCached = true;
  5465. }
  5466. return roxieQueryInfo.getLink();
  5467. }
  5468. static int compareResults(IInterface * const *ll, IInterface * const *rr)
  5469. {
  5470. CLocalWUResult *l = (CLocalWUResult *) *ll;
  5471. CLocalWUResult *r = (CLocalWUResult *) *rr;
  5472. return l->getResultSequence() - r->getResultSequence();
  5473. }
  5474. void CLocalWorkUnit::_loadResults() const
  5475. {
  5476. Owned<IPropertyTreeIterator> r = p->getElements("Results/Result");
  5477. for (r->first(); r->isValid(); r->next())
  5478. {
  5479. IPropertyTree *rp = &r->query();
  5480. rp->Link();
  5481. results.append(*new CLocalWUResult(rp));
  5482. }
  5483. }
  5484. void CLocalWorkUnit::loadResults() const
  5485. {
  5486. CriticalBlock block(crit);
  5487. if (!resultsCached)
  5488. {
  5489. assertex(results.length() == 0);
  5490. _loadResults();
  5491. results.sort(compareResults);
  5492. resultsCached = true;
  5493. }
  5494. }
  5495. void CLocalWorkUnit::loadVariables() const
  5496. {
  5497. CriticalBlock block(crit);
  5498. if (!variablesCached)
  5499. {
  5500. assertex(variables.length() == 0);
  5501. Owned<IPropertyTreeIterator> r = p->getElements("Variables/Variable");
  5502. for (r->first(); r->isValid(); r->next())
  5503. {
  5504. IPropertyTree *rp = &r->query();
  5505. rp->Link();
  5506. variables.append(*new CLocalWUResult(rp));
  5507. }
  5508. variablesCached = true;
  5509. }
  5510. }
  5511. void CLocalWorkUnit::loadTemporaries() const
  5512. {
  5513. CriticalBlock block(crit);
  5514. if (!temporariesCached)
  5515. {
  5516. assertex(temporaries.length() == 0);
  5517. Owned<IPropertyTreeIterator> r = p->getElements("Temporaries/Variable");
  5518. for (r->first(); r->isValid(); r->next())
  5519. {
  5520. IPropertyTree *rp = &r->query();
  5521. rp->Link();
  5522. temporaries.append(*new CLocalWUResult(rp));
  5523. }
  5524. temporariesCached = true;
  5525. }
  5526. }
  5527. void CLocalWorkUnit::deleteTemporaries()
  5528. {
  5529. CriticalBlock block(crit);
  5530. if (temporariesCached)
  5531. {
  5532. temporaries.kill();
  5533. temporariesCached = false;
  5534. }
  5535. p->removeProp("Temporaries");
  5536. }
  5537. IWUResult* CLocalWorkUnit::createResult()
  5538. {
  5539. CriticalBlock block(crit);
  5540. // For this to be legally called, we must have the write-able interface. So we are already locked for write.
  5541. loadResults();
  5542. if (!results.length())
  5543. p->addPropTree("Results", createPTree("Results"));
  5544. IPropertyTree *r = p->queryPropTree("Results");
  5545. IPropertyTree *s = r->addPropTree("Result", createPTree());
  5546. s->Link();
  5547. IWUResult* q = new CLocalWUResult(s);
  5548. q->Link();
  5549. results.append(*q);
  5550. return q;
  5551. }
  5552. IWUResult* CLocalWorkUnit::updateResultByName(const char *qname)
  5553. {
  5554. CriticalBlock block(crit);
  5555. IConstWUResult *existing = getResultByName(qname);
  5556. if (existing)
  5557. return (IWUResult *) existing;
  5558. IWUResult* q = createResult();
  5559. q->setResultName(qname);
  5560. return q;
  5561. }
  5562. IWUResult* CLocalWorkUnit::updateResultBySequence(unsigned seq)
  5563. {
  5564. CriticalBlock block(crit);
  5565. IConstWUResult *existing = getResultBySequence(seq);
  5566. if (existing)
  5567. return (IWUResult *) existing;
  5568. IWUResult* q = createResult();
  5569. q->setResultSequence(seq);
  5570. return q;
  5571. }
  5572. IConstWUResultIterator& CLocalWorkUnit::getResults() const
  5573. {
  5574. CriticalBlock block(crit);
  5575. loadResults();
  5576. return *new CArrayIteratorOf<IConstWUResult,IConstWUResultIterator> (results, 0, (IConstWorkUnit *) this);
  5577. }
  5578. IConstWUResult* CLocalWorkUnit::getResultByName(const char *qname) const
  5579. {
  5580. CriticalBlock block(crit);
  5581. loadResults();
  5582. ForEachItemIn(idx, results)
  5583. {
  5584. SCMStringBuffer name;
  5585. IConstWUResult &cur = results.item(idx);
  5586. cur.getResultName(name);
  5587. if (stricmp(name.str(), qname)==0)
  5588. {
  5589. cur.Link();
  5590. return &cur;
  5591. }
  5592. }
  5593. return NULL;
  5594. }
  5595. IConstWUResult* CLocalWorkUnit::getResultBySequence(unsigned seq) const
  5596. {
  5597. CriticalBlock block(crit);
  5598. loadResults();
  5599. ForEachItemIn(idx, results)
  5600. {
  5601. IConstWUResult &cur = results.item(idx);
  5602. if (cur.getResultSequence() == seq)
  5603. {
  5604. cur.Link();
  5605. return &cur;
  5606. }
  5607. }
  5608. return NULL;
  5609. }
  5610. IConstWUResultIterator& CLocalWorkUnit::getVariables() const
  5611. {
  5612. CriticalBlock block(crit);
  5613. loadVariables();
  5614. return *new CArrayIteratorOf<IConstWUResult,IConstWUResultIterator> (variables, 0, (IConstWorkUnit *) this);
  5615. }
  5616. IConstWUResult* CLocalWorkUnit::getGlobalByName(const char *qname) const
  5617. {
  5618. CriticalBlock block(crit);
  5619. if (strcmp(p->queryName(), GLOBAL_WORKUNIT)==0)
  5620. return getVariableByName(qname);
  5621. Owned <IWorkUnit> global = factory->getGlobalWorkUnit(secMgr, secUser);
  5622. return global->getVariableByName(qname);
  5623. }
  5624. IWUResult* CLocalWorkUnit::updateGlobalByName(const char *qname)
  5625. {
  5626. CriticalBlock block(crit);
  5627. if (strcmp(p->queryName(), GLOBAL_WORKUNIT)==0)
  5628. return updateVariableByName(qname);
  5629. Owned <IWorkUnit> global = factory->getGlobalWorkUnit(secMgr, secUser);
  5630. return global->updateVariableByName(qname);
  5631. }
  5632. IConstWUResult* CLocalWorkUnit::getVariableByName(const char *qname) const
  5633. {
  5634. CriticalBlock block(crit);
  5635. loadVariables();
  5636. ForEachItemIn(idx, variables)
  5637. {
  5638. SCMStringBuffer name;
  5639. IConstWUResult &cur = variables.item(idx);
  5640. cur.getResultName(name);
  5641. if (stricmp(name.str(), qname)==0)
  5642. {
  5643. cur.Link();
  5644. return &cur;
  5645. }
  5646. }
  5647. return NULL;
  5648. }
  5649. IConstWUResult* CLocalWorkUnit::getTemporaryByName(const char *qname) const
  5650. {
  5651. CriticalBlock block(crit);
  5652. loadTemporaries();
  5653. ForEachItemIn(idx, temporaries)
  5654. {
  5655. SCMStringBuffer name;
  5656. IConstWUResult &cur = temporaries.item(idx);
  5657. cur.getResultName(name);
  5658. if (stricmp(name.str(), qname)==0)
  5659. {
  5660. cur.Link();
  5661. return &cur;
  5662. }
  5663. }
  5664. return NULL;
  5665. }
  5666. IConstWUResultIterator& CLocalWorkUnit::getTemporaries() const
  5667. {
  5668. CriticalBlock block(crit);
  5669. loadTemporaries();
  5670. return *new CArrayIteratorOf<IConstWUResult,IConstWUResultIterator> (temporaries, 0, (IConstWorkUnit *) this);
  5671. }
  5672. IWUResult* CLocalWorkUnit::updateTemporaryByName(const char *qname)
  5673. {
  5674. CriticalBlock block(crit);
  5675. IConstWUResult *existing = getTemporaryByName(qname);
  5676. if (existing)
  5677. return (IWUResult *) existing;
  5678. if (!temporaries.length())
  5679. p->addPropTree("Temporaries", createPTree("Temporaries"));
  5680. IPropertyTree *vars = p->queryPropTree("Temporaries");
  5681. IPropertyTree *s = vars->addPropTree("Variable", createPTree("Variable"));
  5682. s->Link();
  5683. IWUResult* q = new CLocalWUResult(s);
  5684. q->Link();
  5685. temporaries.append(*q);
  5686. q->setResultName(qname);
  5687. return q;
  5688. }
  5689. IWUResult* CLocalWorkUnit::updateVariableByName(const char *qname)
  5690. {
  5691. CriticalBlock block(crit);
  5692. IConstWUResult *existing = getVariableByName(qname);
  5693. if (existing)
  5694. return (IWUResult *) existing;
  5695. if (!variables.length())
  5696. p->addPropTree("Variables", createPTree("Variables"));
  5697. IPropertyTree *vars = p->queryPropTree("Variables");
  5698. IPropertyTree *s = vars->addPropTree("Variable", createPTree("Variable"));
  5699. s->Link();
  5700. IWUResult* q = new CLocalWUResult(s);
  5701. q->Link();
  5702. variables.append(*q);
  5703. q->setResultName(qname);
  5704. return q;
  5705. }
  5706. void CLocalWorkUnit::deleteTempFiles(const char *graph, bool deleteOwned, bool deleteJobOwned)
  5707. {
  5708. CriticalBlock block(crit);
  5709. IPropertyTree *files = p->queryPropTree("Files");
  5710. if (!files) return;
  5711. Owned<IPropertyTreeIterator> iter = files->getElements("File");
  5712. ICopyArrayOf<IPropertyTree> toRemove;
  5713. ForEach (*iter)
  5714. {
  5715. IPropertyTree &file = iter->query();
  5716. WUFileKind fileKind = (WUFileKind) file.getPropInt("@kind", WUFileStandard);
  5717. if(file.getPropBool("@temporary")) fileKind = WUFileTemporary; // @temporary, legacy check
  5718. bool needDelete;
  5719. switch(fileKind)
  5720. {
  5721. case WUFileTemporary:
  5722. if(graph==NULL)
  5723. needDelete = true;
  5724. else
  5725. {
  5726. const char *graphOwner = file.queryProp("@graph");
  5727. needDelete = ((graphOwner==NULL) || (strcmp(graph, graphOwner)==0));
  5728. }
  5729. break;
  5730. case WUFileJobOwned:
  5731. needDelete = ((graph==NULL) && deleteJobOwned);
  5732. break;
  5733. case WUFileOwned:
  5734. needDelete = ((graph==NULL) && deleteOwned);
  5735. break;
  5736. default:
  5737. needDelete = false;
  5738. }
  5739. if(needDelete)
  5740. {
  5741. const char *name = file.queryProp("@name");
  5742. LOG(MCdebugProgress, unknownJob, "Removing workunit file %s from DFS", name);
  5743. queryDistributedFileDirectory().removeEntry(name, queryUserDescriptor());
  5744. toRemove.append(file);
  5745. }
  5746. }
  5747. ForEachItemIn(r, toRemove) files->removeTree(&toRemove.item(r));
  5748. }
  5749. static void _noteFileRead(IDistributedFile *file, IPropertyTree *filesRead)
  5750. {
  5751. IDistributedSuperFile *super = file->querySuperFile();
  5752. StringBuffer fname;
  5753. file->getLogicalName(fname);
  5754. StringBuffer path("File[@name=\"");
  5755. path.append(fname).append("\"]");
  5756. IPropertyTree *fileTree = filesRead->queryPropTree(path.str());
  5757. if (fileTree)
  5758. fileTree->setPropInt("@useCount", fileTree->getPropInt("@useCount")+1);
  5759. else
  5760. {
  5761. StringBuffer cluster;
  5762. file->getClusterName(0,cluster);
  5763. fileTree = createPTree();
  5764. fileTree->setProp("@name", fname.str());
  5765. fileTree->setProp("@cluster", cluster.str());
  5766. fileTree->setPropInt("@useCount", 1);
  5767. fileTree = filesRead->addPropTree("File", fileTree);
  5768. }
  5769. if (super)
  5770. {
  5771. Owned<IDistributedFileIterator> iter = super->getSubFileIterator(false);
  5772. ForEach (*iter)
  5773. {
  5774. IDistributedFile &file = iter->query();
  5775. StringBuffer fname;
  5776. file.getLogicalName(fname);
  5777. Owned<IPropertyTree> subfile = createPTree();
  5778. subfile->setProp("@name", fname.str());
  5779. fileTree->addPropTree("Subfile", subfile.getClear());
  5780. _noteFileRead(&file, filesRead);
  5781. }
  5782. }
  5783. }
  5784. void CLocalWorkUnit::noteFileRead(IDistributedFile *file)
  5785. {
  5786. CriticalBlock block(crit);
  5787. IPropertyTree *files = p->queryPropTree("FilesRead");
  5788. if (!files)
  5789. files = p->addPropTree("FilesRead", createPTree());
  5790. _noteFileRead(file, files);
  5791. }
  5792. static void addFile(IPropertyTree *files, const char *fileName, const char *cluster, unsigned usageCount, WUFileKind fileKind, const char *graphOwner)
  5793. {
  5794. StringBuffer path("File[@name=\"");
  5795. path.append(fileName).append("\"]");
  5796. if (cluster)
  5797. path.append("[@cluster=\"").append(cluster).append("\"]");
  5798. IPropertyTree *file = files->queryPropTree(path.str());
  5799. if (file) files->removeTree(file);
  5800. file = createPTree();
  5801. file->setProp("@name", fileName);
  5802. if (cluster)
  5803. file->setProp("@cluster", cluster);
  5804. if (graphOwner)
  5805. file->setProp("@graph", graphOwner);
  5806. file->setPropInt("@kind", (unsigned)fileKind);
  5807. if (WUFileTemporary == fileKind)
  5808. file->setPropInt("@usageCount", usageCount);
  5809. files->addPropTree("File", file);
  5810. }
  5811. void CLocalWorkUnit::addFile(const char *fileName, StringArray *clusters, unsigned usageCount, WUFileKind fileKind, const char *graphOwner)
  5812. {
  5813. CriticalBlock block(crit);
  5814. IPropertyTree *files = p->queryPropTree("Files");
  5815. if (!files)
  5816. files = p->addPropTree("Files", createPTree());
  5817. if (!clusters)
  5818. addFile(fileName, NULL, usageCount, fileKind, graphOwner);
  5819. else
  5820. {
  5821. ForEachItemIn(c, *clusters)
  5822. ::addFile(files, fileName, clusters->item(c), usageCount, fileKind, graphOwner);
  5823. }
  5824. }
  5825. void CLocalWorkUnit::releaseFile(const char *fileName)
  5826. {
  5827. StringBuffer path("File[@name=\"");
  5828. path.append(fileName).append("\"]");
  5829. CriticalBlock block(crit);
  5830. IPropertyTree *files = p->queryPropTree("Files");
  5831. if (!files) return;
  5832. Owned<IPropertyTreeIterator> fiter = files->getElements(path.str());
  5833. ForEach (*fiter)
  5834. {
  5835. IPropertyTree *file = &fiter->query();
  5836. unsigned usageCount = file->getPropInt("@usageCount");
  5837. if (usageCount > 1)
  5838. file->setPropInt("@usageCount", usageCount-1);
  5839. else
  5840. {
  5841. StringAttr name(file->queryProp("@name"));
  5842. files->removeTree(file);
  5843. if (!name.isEmpty()&&(1 == usageCount))
  5844. {
  5845. if (queryDistributedFileDirectory().removeEntry(fileName, queryUserDescriptor()))
  5846. LOG(MCdebugProgress, unknownJob, "Removed (released) file %s from DFS", name.get());
  5847. }
  5848. }
  5849. }
  5850. }
  5851. void CLocalWorkUnit::clearGraphProgress()
  5852. {
  5853. CConstGraphProgress::deleteWuidProgress(p->queryName());
  5854. }
  5855. void CLocalWorkUnit::resetBeforeGeneration()
  5856. {
  5857. CriticalBlock block(crit);
  5858. //Remove all associated files
  5859. Owned<IWUQuery> q = updateQuery();
  5860. q->removeAssociatedFiles();
  5861. //Remove any pre-existing workflow information
  5862. workflowIterator.clear();
  5863. p->removeProp("Workflow");
  5864. }
  5865. unsigned CLocalWorkUnit::queryFileUsage(const char *fileName) const
  5866. {
  5867. StringBuffer path("Files/File[@name=\"");
  5868. path.append(fileName).append("\"]/@usageCount");
  5869. CriticalBlock block(crit);
  5870. return p->getPropInt(path.str());
  5871. }
  5872. IPropertyTree *CLocalWorkUnit::getDiskUsageStats()
  5873. {
  5874. return p->getPropTree("DiskUsageStats");
  5875. }
  5876. void CLocalWorkUnit::addDiskUsageStats(__int64 _avgNodeUsage, unsigned _minNode, __int64 _minNodeUsage, unsigned _maxNode, __int64 _maxNodeUsage, __int64 _graphId)
  5877. {
  5878. IPropertyTree *stats = p->queryPropTree("DiskUsageStats");
  5879. offset_t maxNodeUsage;
  5880. if (stats)
  5881. maxNodeUsage = stats->getPropInt64("@maxNodeUsage");
  5882. else
  5883. {
  5884. stats = p->addPropTree("DiskUsageStats", createPTree());
  5885. maxNodeUsage = 0;
  5886. }
  5887. if ((offset_t)_maxNodeUsage > maxNodeUsage)
  5888. {
  5889. // record all details at time of max node usage.
  5890. stats->setPropInt("@minNode", _minNode);
  5891. stats->setPropInt("@maxNode", _maxNode);
  5892. stats->setPropInt64("@minNodeUsage", _minNodeUsage);
  5893. stats->setPropInt64("@maxNodeUsage", _maxNodeUsage);
  5894. stats->setPropInt64("@graphId", _graphId);
  5895. if (_avgNodeUsage)
  5896. {
  5897. unsigned _skewHi = (unsigned)((100 * (_maxNodeUsage-_avgNodeUsage))/_avgNodeUsage);
  5898. unsigned _skewLo = (unsigned)((100 * (_avgNodeUsage-_minNodeUsage))/_avgNodeUsage);
  5899. stats->setPropInt("@skewHi", _skewHi);
  5900. stats->setPropInt("@skewLo", _skewLo);
  5901. }
  5902. }
  5903. }
  5904. IPropertyTreeIterator & CLocalWorkUnit::getFileIterator() const
  5905. {
  5906. CriticalBlock block(crit);
  5907. return * p->getElements("Files/File");
  5908. }
  5909. IPropertyTreeIterator & CLocalWorkUnit::getFilesReadIterator() const
  5910. {
  5911. CriticalBlock block(crit);
  5912. return * p->getElements("FilesRead/File");
  5913. }
  5914. //=================================================================================================
  5915. bool CLocalWorkUnit::switchThorQueue(const char *cluster, IQueueSwitcher *qs)
  5916. {
  5917. CriticalBlock block(crit);
  5918. if (qs->isAuto()&&!getAllowAutoQueueSwitch())
  5919. return false;
  5920. Owned<IConstWUClusterInfo> newci = getTargetClusterInfo(cluster);
  5921. if (!newci)
  5922. return false;
  5923. StringBuffer currentcluster;
  5924. if (!p->getProp("@clusterName",currentcluster))
  5925. return false;
  5926. Owned<IConstWUClusterInfo> curci = getTargetClusterInfo(currentcluster.str());
  5927. if (!curci)
  5928. return false;
  5929. SCMStringBuffer curqname;
  5930. curci->getThorQueue(curqname);
  5931. const char *wuid = p->queryName();
  5932. void *qi = qs->getQ(curqname.str(),wuid);
  5933. if (!qi)
  5934. return false;
  5935. setClusterName(cluster);
  5936. SCMStringBuffer newqname;
  5937. newci->getThorQueue(newqname);
  5938. qs->putQ(newqname.str(),wuid,qi);
  5939. return true;
  5940. }
  5941. //=================================================================================================
  5942. IPropertyTree *CLocalWorkUnit::getUnpackedTree(bool includeProgress) const
  5943. {
  5944. Owned<IPropertyTree> ret = createPTreeFromIPT(p);
  5945. Owned<IConstWUGraphIterator> graphIter = &getGraphs(GraphTypeAny);
  5946. ForEach(*graphIter)
  5947. {
  5948. IConstWUGraph &graph = graphIter->query();
  5949. Owned<IPropertyTree> graphTree = graph.getXGMMLTree(includeProgress);
  5950. SCMStringBuffer gName;
  5951. graph.getName(gName);
  5952. StringBuffer xpath("Graphs/Graph[@name=\"");
  5953. xpath.append(gName.s).append("\"]/xgmml");
  5954. IPropertyTree *xgmml = ret->queryPropTree(xpath.str());
  5955. if (xgmml) // don't know of any reason it shouldn't exist
  5956. {
  5957. xgmml->removeProp("graphBin");
  5958. xgmml->setPropTree("graph", graphTree.getClear());
  5959. }
  5960. }
  5961. return ret.getClear();
  5962. }
  5963. void CLocalWorkUnit::loadGraphs() const
  5964. {
  5965. CriticalBlock block(crit);
  5966. if (!cachedGraphs.get())
  5967. {
  5968. MemoryBuffer buf;
  5969. IPropertyTree *t = p->queryPropTree("PackedGraphs");
  5970. if (t&&t->getPropBin(NULL,buf)) {
  5971. cachedGraphs.setown(createPTree(buf));
  5972. }
  5973. else
  5974. cachedGraphs.set(p->queryPropTree("Graphs"));
  5975. if (cachedGraphs.get())
  5976. {
  5977. Owned<IPropertyTreeIterator> iter = cachedGraphs->getElements("Graph");
  5978. ForEach(*iter)
  5979. {
  5980. IPropertyTree &graph = iter->query();
  5981. graphs.append(*new CLocalWUGraph(*this, LINK(&graph)));
  5982. }
  5983. }
  5984. }
  5985. }
  5986. mapEnums graphTypes[] = {
  5987. { GraphTypeAny, "unknown" },
  5988. { GraphTypeProgress, "progress" },
  5989. { GraphTypeEcl, "ECL" },
  5990. { GraphTypeActivities, "activities" },
  5991. { GraphTypeSubProgress, "subgraph" },
  5992. { GraphTypeSize, NULL },
  5993. };
  5994. CLocalWUGraph::CLocalWUGraph(const CLocalWorkUnit &_owner, IPropertyTree *props) : p(props), owner(_owner)
  5995. {
  5996. wuidVersion = owner.getWuidVersion();
  5997. }
  5998. IStringVal& CLocalWUGraph::getName(IStringVal &str) const
  5999. {
  6000. str.set(p->queryProp("@name"));
  6001. return str;
  6002. }
  6003. IStringVal& CLocalWUGraph::getLabel(IStringVal &str) const
  6004. {
  6005. if (wuidVersion >= 2)
  6006. {
  6007. str.set(p->queryProp("@label"));
  6008. return str;
  6009. }
  6010. else
  6011. {
  6012. Owned<IPropertyTree> xgmml = getXGMMLTree(false);
  6013. str.set(xgmml->queryProp("@label"));
  6014. return str;
  6015. }
  6016. }
  6017. WUGraphState CLocalWUGraph::getState() const
  6018. {
  6019. Owned<IConstWUGraphProgress> graphProgress = owner.getGraphProgress(p->queryProp("@name"));
  6020. if (!graphProgress)
  6021. return WUGraphUnknown;
  6022. return graphProgress->queryGraphState();
  6023. }
  6024. IStringVal& CLocalWUGraph::getXGMML(IStringVal &str, bool mergeProgress) const
  6025. {
  6026. Owned<IPropertyTree> xgmml = getXGMMLTree(mergeProgress);
  6027. if (xgmml)
  6028. {
  6029. StringBuffer x;
  6030. toXML(xgmml, x);
  6031. str.set(x.str());
  6032. }
  6033. return str;
  6034. }
  6035. unsigned CLocalWorkUnit::getGraphCount() const
  6036. {
  6037. CriticalBlock block(crit);
  6038. if (p->hasProp("Graphs"))
  6039. {
  6040. return p->queryPropTree("Graphs")->numChildren();
  6041. }
  6042. return 0;
  6043. }
  6044. unsigned CLocalWorkUnit::getSourceFileCount() const
  6045. {
  6046. CriticalBlock block(crit);
  6047. if (p->hasProp("FilesRead"))
  6048. {
  6049. return p->queryPropTree("FilesRead")->numChildren();
  6050. }
  6051. return 0;
  6052. }
  6053. unsigned CLocalWorkUnit::getResultCount() const
  6054. {
  6055. CriticalBlock block(crit);
  6056. if (p->hasProp("Results"))
  6057. {
  6058. return p->queryPropTree("Results")->numChildren();
  6059. }
  6060. return 0;
  6061. }
  6062. unsigned CLocalWorkUnit::getVariableCount() const
  6063. {
  6064. CriticalBlock block(crit);
  6065. if (p->hasProp("Variables"))
  6066. {
  6067. return p->queryPropTree("Variables")->numChildren();
  6068. }
  6069. return 0;
  6070. }
  6071. unsigned CLocalWorkUnit::getApplicationValueCount() const
  6072. {
  6073. CriticalBlock block(crit);
  6074. if (p->hasProp("Application"))
  6075. {
  6076. return p->queryPropTree("Application")->numChildren();
  6077. }
  6078. return 0;
  6079. }
  6080. StringBuffer &appendPTreeOpenTag(StringBuffer &s, IPropertyTree *tree, const char *name, unsigned indent)
  6081. {
  6082. appendXMLOpenTag(s, name, NULL, false);
  6083. Owned<IAttributeIterator> attrs = tree->getAttributes(true);
  6084. if (attrs->first())
  6085. {
  6086. unsigned attributeindent = indent + (size32_t) strlen(name);
  6087. unsigned count = attrs->count();
  6088. bool doindent = false;
  6089. ForEach(*attrs)
  6090. {
  6091. if (doindent)
  6092. s.append('\n').appendN(attributeindent, ' ');
  6093. else if (count > 3)
  6094. doindent = true;
  6095. appendXMLAttr(s, attrs->queryName()+1, attrs->queryValue());
  6096. }
  6097. }
  6098. s.append('>');
  6099. return s;
  6100. }
  6101. IStringVal &CLocalWorkUnit::getXmlParams(IStringVal &str, bool hidePasswords) const
  6102. {
  6103. CriticalBlock block(crit);
  6104. IPropertyTree *paramTree = p->queryPropTree("Parameters");
  6105. if (!paramTree)
  6106. return str;
  6107. StringBuffer xml;
  6108. if (!hidePasswords)
  6109. toXML(paramTree, xml);
  6110. else
  6111. {
  6112. appendPTreeOpenTag(xml.append(' '), paramTree, "Parameters", 0).append('\n');
  6113. Owned<IPropertyTreeIterator> elems = paramTree->getElements("*");
  6114. ForEach(*elems)
  6115. {
  6116. const char *paramname = elems->query().queryName();
  6117. VStringBuffer xpath("Variables/Variable[@name='%s']/Format/@password", paramname);
  6118. if (p->getPropBool(xpath))
  6119. appendXMLTag(xml.append(" "), paramname, "***").append('\n');
  6120. else
  6121. toXML(&elems->query(), xml, 2);
  6122. }
  6123. appendXMLCloseTag(xml.append(' '), "Parameters").append('\n');
  6124. }
  6125. str.set(xml);
  6126. return str;
  6127. }
  6128. const IPropertyTree *CLocalWorkUnit::getXmlParams() const
  6129. {
  6130. CriticalBlock block(crit);
  6131. return p->getPropTree("Parameters");
  6132. }
  6133. void CLocalWorkUnit::setXmlParams(const char *params)
  6134. {
  6135. CriticalBlock block(crit);
  6136. p->setPropTree("Parameters", createPTreeFromXMLString(params));
  6137. }
  6138. void CLocalWorkUnit::setXmlParams(IPropertyTree *tree)
  6139. {
  6140. CriticalBlock block(crit);
  6141. p->setPropTree("Parameters", tree);
  6142. }
  6143. unsigned __int64 CLocalWorkUnit::getHash() const
  6144. {
  6145. CriticalBlock block(crit);
  6146. return p->getPropInt64("@hash");
  6147. }
  6148. void CLocalWorkUnit::setHash(unsigned __int64 hash)
  6149. {
  6150. CriticalBlock block(crit);
  6151. p->setPropInt64("@hash", hash);
  6152. }
  6153. IConstWUGraphMetaIterator& CLocalWorkUnit::getGraphsMeta(WUGraphType type) const
  6154. {
  6155. /* NB: this method should be 'cheap', loadGraphs() creates IConstWUGraph interfaces to the graphs
  6156. * it does not actually pull the graph data. We only use IConstWUGraphMeta here, which never probes the xgmml
  6157. * This method also connects to the graph progress (/GraphProgress/<wuid>) using a single connection, in order
  6158. * to get state information, it does not pull all the progress data.
  6159. */
  6160. CriticalBlock block(crit);
  6161. loadGraphs();
  6162. class CConstWUGraphMetaIterator: public CInterface, implements IConstWUGraphMetaIterator, implements IConstWUGraphMeta
  6163. {
  6164. StringAttr wuid;
  6165. WUGraphType type;
  6166. Owned<IConstWUGraphIterator> graphIter;
  6167. IConstWUGraph *curGraph;
  6168. Owned<IConstWUGraphProgress> curGraphProgress;
  6169. Owned<IRemoteConnection> progressConn;
  6170. bool match()
  6171. {
  6172. return (GraphTypeAny == type) || (type == graphIter->query().getType());
  6173. }
  6174. void setCurrent(IConstWUGraph &graph)
  6175. {
  6176. curGraph = &graph;
  6177. SCMStringBuffer graphName;
  6178. curGraph->getName(graphName);
  6179. if (progressConn)
  6180. {
  6181. IPropertyTree *progress = progressConn->queryRoot()->queryPropTree(graphName.str());
  6182. if (progress)
  6183. {
  6184. curGraphProgress.setown(new CConstGraphProgress(wuid, graphName.str(), progress));
  6185. return;
  6186. }
  6187. }
  6188. curGraphProgress.clear();
  6189. }
  6190. public:
  6191. IMPLEMENT_IINTERFACE;
  6192. CConstWUGraphMetaIterator(const char *_wuid, IConstWUGraphIterator *_graphIter, WUGraphType _type)
  6193. : wuid(_wuid), graphIter(_graphIter), type(_type)
  6194. {
  6195. curGraph = NULL;
  6196. StringBuffer progressPath;
  6197. progressPath.append("/GraphProgress/").append(wuid);
  6198. progressConn.setown(querySDS().connect(progressPath.str(), myProcessSession(), RTM_NONE, SDS_LOCK_TIMEOUT));
  6199. }
  6200. virtual bool first()
  6201. {
  6202. curGraph = NULL;
  6203. curGraphProgress.clear();
  6204. if (!graphIter->first())
  6205. return false;
  6206. if (match())
  6207. {
  6208. setCurrent(graphIter->query());
  6209. return true;
  6210. }
  6211. return next();
  6212. }
  6213. virtual bool next()
  6214. {
  6215. while (graphIter->next())
  6216. {
  6217. if (match())
  6218. {
  6219. setCurrent(graphIter->query());
  6220. return true;
  6221. }
  6222. }
  6223. curGraph = NULL;
  6224. curGraphProgress.clear();
  6225. return false;
  6226. }
  6227. virtual bool isValid()
  6228. {
  6229. return NULL != curGraph;
  6230. }
  6231. virtual IConstWUGraphMeta & query()
  6232. {
  6233. return *this;
  6234. }
  6235. // IConstWUGraphMeta
  6236. virtual IStringVal & getName(IStringVal & ret) const
  6237. {
  6238. return curGraph->getName(ret);
  6239. }
  6240. virtual IStringVal & getLabel(IStringVal & ret) const
  6241. {
  6242. return curGraph->getLabel(ret);
  6243. }
  6244. virtual IStringVal & getTypeName(IStringVal & ret) const
  6245. {
  6246. return curGraph->getTypeName(ret);
  6247. }
  6248. virtual WUGraphType getType() const
  6249. {
  6250. return curGraph->getType();
  6251. }
  6252. virtual WUGraphState getState() const
  6253. {
  6254. if (!curGraphProgress)
  6255. return WUGraphUnknown;
  6256. return curGraphProgress->queryGraphState();
  6257. }
  6258. };
  6259. IConstWUGraphIterator *graphIter = new CArrayIteratorOf<IConstWUGraph,IConstWUGraphIterator> (graphs, 0, (IConstWorkUnit *) this);
  6260. return * new CConstWUGraphMetaIterator(p->queryName(), graphIter, type);
  6261. }
  6262. IConstWUGraphIterator& CLocalWorkUnit::getGraphs(WUGraphType type) const
  6263. {
  6264. CriticalBlock block(crit);
  6265. loadGraphs();
  6266. IConstWUGraphIterator *giter = new CArrayIteratorOf<IConstWUGraph,IConstWUGraphIterator> (graphs, 0, (IConstWorkUnit *) this);
  6267. if (type!=GraphTypeAny) {
  6268. class CConstWUGraphIterator: public CInterface, implements IConstWUGraphIterator
  6269. {
  6270. WUGraphType type;
  6271. Owned<IConstWUGraphIterator> base;
  6272. bool match()
  6273. {
  6274. return base->query().getType()==type;
  6275. }
  6276. public:
  6277. IMPLEMENT_IINTERFACE;
  6278. CConstWUGraphIterator(IConstWUGraphIterator *_base,WUGraphType _type)
  6279. : base(_base)
  6280. {
  6281. type = _type;
  6282. }
  6283. bool first()
  6284. {
  6285. if (!base->first())
  6286. return false;
  6287. if (match())
  6288. return true;
  6289. return next();
  6290. }
  6291. bool next()
  6292. {
  6293. while (base->next())
  6294. if (match())
  6295. return true;
  6296. return false;
  6297. }
  6298. virtual bool isValid()
  6299. {
  6300. return base->isValid();
  6301. }
  6302. IConstWUGraph & query()
  6303. {
  6304. return base->query();
  6305. }
  6306. };
  6307. giter = new CConstWUGraphIterator(giter,type);
  6308. }
  6309. return *giter;
  6310. }
  6311. IConstWUGraph* CLocalWorkUnit::getGraph(const char *qname) const
  6312. {
  6313. CriticalBlock block(crit);
  6314. loadGraphs();
  6315. ForEachItemIn(idx, graphs)
  6316. {
  6317. SCMStringBuffer name;
  6318. IConstWUGraph &cur = graphs.item(idx);
  6319. cur.getName(name);
  6320. if (stricmp(name.str(), qname)==0)
  6321. {
  6322. cur.Link();
  6323. return &cur;
  6324. }
  6325. }
  6326. return NULL;
  6327. }
  6328. IWUGraph* CLocalWorkUnit::createGraph()
  6329. {
  6330. // For this to be legally called, we must have the write-able interface. So we are already locked for write.
  6331. CriticalBlock block(crit);
  6332. ensureGraphsUnpacked();
  6333. loadGraphs();
  6334. if (!graphs.ordinality())
  6335. p->addPropTree("Graphs", createPTree("Graphs"));
  6336. IPropertyTree *r = p->queryPropTree("Graphs");
  6337. IPropertyTree *s = r->addPropTree("Graph", createPTree());
  6338. s->Link();
  6339. IWUGraph* q = new CLocalWUGraph(*this, s);
  6340. q->Link();
  6341. graphs.append(*q);
  6342. return q;
  6343. }
  6344. IWUGraph * CLocalWorkUnit::createGraph(const char * name, WUGraphType type, IPropertyTree *xgmml)
  6345. {
  6346. CriticalBlock block(crit);
  6347. ensureGraphsUnpacked();
  6348. Linked<IConstWUGraph> existing = getGraph(name);
  6349. if (existing)
  6350. throwUnexpected();
  6351. if (!graphs.length())
  6352. p->addPropTree("Graphs", createPTree("Graphs"));
  6353. IPropertyTree *r = p->queryPropTree("Graphs");
  6354. IPropertyTree *s = r->addPropTree("Graph", createPTree());
  6355. IWUGraph* q = new CLocalWUGraph(*this, LINK(s));
  6356. graphs.append(*LINK(q));
  6357. q->setName(name);
  6358. q->setXGMMLTree(xgmml);
  6359. q->setType(type);
  6360. return q;
  6361. }
  6362. IWUGraph * CLocalWorkUnit::updateGraph(const char * name)
  6363. {
  6364. CriticalBlock block(crit);
  6365. ensureGraphsUnpacked();
  6366. return (IWUGraph *)getGraph(name);
  6367. }
  6368. IConstWUGraphProgress *CLocalWorkUnit::getGraphProgress(const char *name) const
  6369. {
  6370. CriticalBlock block(crit);
  6371. return new CConstGraphProgress(p->queryName(), name);
  6372. }
  6373. void CLocalWUGraph::setName(const char *str)
  6374. {
  6375. p->setProp("@name", str);
  6376. progress.clear();
  6377. progress.setown(new CConstGraphProgress(owner.queryWuid(), str));
  6378. }
  6379. void CLocalWUGraph::setLabel(const char *str)
  6380. {
  6381. p->setProp("@label", str);
  6382. }
  6383. void CLocalWUGraph::setXGMML(const char *str)
  6384. {
  6385. setXGMMLTree(createPTreeFromXMLString(str));
  6386. }
  6387. void CLocalWUGraph::setXGMMLTree(IPropertyTree *_graph, bool compress)
  6388. {
  6389. assertex(strcmp(_graph->queryName(), "graph")==0);
  6390. IPropertyTree *xgmml = p->setPropTree("xgmml", createPTree());
  6391. if (compress)
  6392. {
  6393. MemoryBuffer mb;
  6394. _graph->serialize(mb);
  6395. xgmml->setPropBin("graphBin", mb.length(), mb.toByteArray());
  6396. graph.setown(_graph);
  6397. }
  6398. else
  6399. xgmml->setPropTree("graph", _graph);
  6400. }
  6401. static void expandAttributes(IPropertyTree & targetNode, IPropertyTree & progressNode)
  6402. {
  6403. Owned<IAttributeIterator> aIter = progressNode.getAttributes();
  6404. ForEach (*aIter)
  6405. {
  6406. const char *aName = aIter->queryName()+1;
  6407. if (0 != stricmp("id", aName)) // "id" reserved.
  6408. {
  6409. IPropertyTree *att = targetNode.addPropTree("att", createPTree());
  6410. att->setProp("@name", aName);
  6411. att->setProp("@value", aIter->queryValue());
  6412. }
  6413. }
  6414. }
  6415. void CLocalWUGraph::mergeProgress(IPropertyTree &rootNode, IPropertyTree &progressTree, const unsigned &progressV) const
  6416. {
  6417. IPropertyTree *graphNode = rootNode.queryPropTree("att/graph");
  6418. if (!graphNode) return;
  6419. unsigned nodeId = rootNode.getPropInt("@id");
  6420. StringBuffer progressNodePath("node[@id=\"");
  6421. progressNodePath.append(nodeId).append("\"]");
  6422. IPropertyTree *progressNode = progressTree.queryPropTree(progressNodePath.str());
  6423. if (progressNode)
  6424. {
  6425. expandAttributes(*graphNode, *progressNode);
  6426. Owned<IPropertyTreeIterator> edges = progressNode->getElements("edge");
  6427. ForEach (*edges)
  6428. {
  6429. IPropertyTree &edge = edges->query();
  6430. StringBuffer edgePath("edge[@id=\"");
  6431. edgePath.append(edge.queryProp("@id")).append("\"]");
  6432. IPropertyTree *graphEdge = graphNode->queryPropTree(edgePath.str());
  6433. if (graphEdge)
  6434. {
  6435. if (progressV < 1)
  6436. mergePTree(graphEdge, &edge);
  6437. else
  6438. { // must translate to XGMML format
  6439. expandAttributes(*graphEdge, edge);
  6440. // This is really only here, so that our progress format can use non-attribute values, which have different efficiency qualifies (e.g. can be external by dali)
  6441. Owned<IPropertyTreeIterator> iter = edge.getElements("*");
  6442. ForEach (*iter)
  6443. {
  6444. IPropertyTree &t = iter->query();
  6445. IPropertyTree *att = graphEdge->addPropTree("att", createPTree());
  6446. att->setProp("@name", t.queryName());
  6447. att->setProp("@value", t.queryProp(NULL));
  6448. }
  6449. }
  6450. }
  6451. }
  6452. Owned<IPropertyTreeIterator> nodes = progressNode->getElements("node");
  6453. ForEach (*nodes)
  6454. {
  6455. IPropertyTree &node = nodes->query();
  6456. StringBuffer nodePath("node[@id=\"");
  6457. nodePath.append(node.queryProp("@id")).append("\"]");
  6458. IPropertyTree *_node = graphNode->queryPropTree(nodePath.str());
  6459. if (_node)
  6460. {
  6461. if (progressV < 1)
  6462. mergePTree(_node, &node);
  6463. else
  6464. { // must translate to XGMML format
  6465. expandAttributes(*_node, node);
  6466. }
  6467. }
  6468. }
  6469. }
  6470. Owned<IPropertyTreeIterator> iter = graphNode->getElements("node");
  6471. ForEach (*iter)
  6472. mergeProgress(iter->query(), progressTree, progressV);
  6473. }
  6474. IPropertyTree * CLocalWUGraph::getXGMMLTreeRaw() const
  6475. {
  6476. return p->getPropTree("xgmml");
  6477. }
  6478. IPropertyTree * CLocalWUGraph::getXGMMLTree(bool doMergeProgress) const
  6479. {
  6480. if (!graph)
  6481. {
  6482. // NB: although graphBin introduced in wuidVersion==2,
  6483. // daliadmin can retrospectively compress existing graphs, so need to check for all versions
  6484. MemoryBuffer mb;
  6485. if (p->getPropBin("xgmml/graphBin", mb))
  6486. graph.setown(createPTree(mb));
  6487. else
  6488. graph.setown(p->getBranch("xgmml/graph"));
  6489. if (!graph)
  6490. return NULL;
  6491. }
  6492. if (!doMergeProgress)
  6493. return graph.getLink();
  6494. else
  6495. {
  6496. Owned<IPropertyTree> copy = createPTreeFromIPT(graph);
  6497. Owned<IConstWUGraphProgress> _progress;
  6498. if (progress) _progress.set(progress);
  6499. else
  6500. _progress.setown(new CConstGraphProgress(owner.queryWuid(), p->queryProp("@name")));
  6501. //MORE: Eventually this should directly access the new stats structure
  6502. unsigned progressV = _progress->queryFormatVersion();
  6503. Owned<IPropertyTree> progressTree = _progress->getProgressTree();
  6504. Owned<IPropertyTreeIterator> nodeIterator = copy->getElements("node");
  6505. ForEach (*nodeIterator)
  6506. mergeProgress(nodeIterator->query(), *progressTree, progressV);
  6507. return copy.getClear();
  6508. }
  6509. }
  6510. WUGraphType CLocalWUGraph::getType() const
  6511. {
  6512. return (WUGraphType) getEnum(p, "@type", graphTypes);
  6513. }
  6514. IStringVal & CLocalWUGraph::getTypeName(IStringVal &str) const
  6515. {
  6516. str.set(p->queryProp("@type"));
  6517. if (!str.length())
  6518. str.set("unknown");
  6519. return str;
  6520. }
  6521. void CLocalWUGraph::setType(WUGraphType _type)
  6522. {
  6523. setEnum(p, "@type", _type, graphTypes);
  6524. }
  6525. //=================================================================================================
  6526. mapEnums queryFileTypes[] = {
  6527. { FileTypeCpp, "cpp" },
  6528. { FileTypeDll, "dll" },
  6529. { FileTypeResText, "res" },
  6530. { FileTypeHintXml, "hint" },
  6531. { FileTypeXml, "xml" },
  6532. { FileTypeSize, NULL },
  6533. };
  6534. CLocalWUAssociated::CLocalWUAssociated(IPropertyTree *props) : p(props)
  6535. {
  6536. }
  6537. WUFileType CLocalWUAssociated::getType() const
  6538. {
  6539. return (WUFileType)getEnum(p, "@type", queryFileTypes);
  6540. }
  6541. IStringVal & CLocalWUAssociated::getDescription(IStringVal & str) const
  6542. {
  6543. str.set(p->queryProp("@desc"));
  6544. return str;
  6545. }
  6546. IStringVal & CLocalWUAssociated::getIp(IStringVal & str) const
  6547. {
  6548. str.set(p->queryProp("@ip"));
  6549. return str;
  6550. }
  6551. IStringVal & CLocalWUAssociated::getName(IStringVal & str) const
  6552. {
  6553. str.set(p->queryProp("@filename"));
  6554. return str;
  6555. }
  6556. IStringVal & CLocalWUAssociated::getNameTail(IStringVal & str) const
  6557. {
  6558. str.set(pathTail(p->queryProp("@filename")));
  6559. return str;
  6560. }
  6561. unsigned CLocalWUAssociated::getCrc() const
  6562. {
  6563. return p->getPropInt("@crc", 0);
  6564. }
  6565. //=================================================================================================
  6566. CLocalWUQuery::CLocalWUQuery(IPropertyTree *props) : p(props)
  6567. {
  6568. associatedCached = false;
  6569. }
  6570. mapEnums queryTypes[] = {
  6571. { QueryTypeUnknown, "unknown" },
  6572. { QueryTypeEcl, "ECL" },
  6573. { QueryTypeSql, "SQL" },
  6574. { QueryTypeXml, "XML" },
  6575. { QueryTypeAttribute, "Attribute" },
  6576. { QueryTypeSize, NULL },
  6577. };
  6578. WUQueryType CLocalWUQuery::getQueryType() const
  6579. {
  6580. return (WUQueryType) getEnum(p, "@type", queryTypes);
  6581. }
  6582. void CLocalWUQuery::setQueryType(WUQueryType qt)
  6583. {
  6584. setEnum(p, "@type", qt, queryTypes);
  6585. }
  6586. IStringVal& CLocalWUQuery::getQueryText(IStringVal &str) const
  6587. {
  6588. str.set(p->queryProp("Text"));
  6589. return str;
  6590. }
  6591. IStringVal& CLocalWUQuery::getQueryShortText(IStringVal &str) const
  6592. {
  6593. const char * text = p->queryProp("Text");
  6594. if (isArchiveQuery(text))
  6595. {
  6596. Owned<IPropertyTree> xml = createPTreeFromXMLString(text, ipt_caseInsensitive);
  6597. const char * path = xml->queryProp("Query/@attributePath");
  6598. if (path)
  6599. {
  6600. IPropertyTree * resolved = resolveDefinitionInArchive(xml, path);
  6601. if (resolved)
  6602. str.set(resolved->queryProp(NULL));
  6603. }
  6604. else
  6605. str.set(xml->queryProp("Query"));
  6606. }
  6607. else
  6608. str.set(text);
  6609. return str;
  6610. }
  6611. IStringVal& CLocalWUQuery::getQueryName(IStringVal &str) const
  6612. {
  6613. str.set(p->queryProp("@name"));
  6614. return str;
  6615. }
  6616. IStringVal & CLocalWUQuery::getQueryMainDefinition(IStringVal & str) const
  6617. {
  6618. str.set(p->queryProp("@main"));
  6619. return str;
  6620. }
  6621. IStringVal& CLocalWUQuery::getQueryDllName(IStringVal &str) const
  6622. {
  6623. Owned<IConstWUAssociatedFile> entry = getAssociatedFile(FileTypeDll, 0);
  6624. if (entry)
  6625. entry->getNameTail(str);
  6626. return str;
  6627. }
  6628. IStringVal& CLocalWUQuery::getQueryCppName(IStringVal &str) const
  6629. {
  6630. Owned<IConstWUAssociatedFile> entry = getAssociatedFile(FileTypeCpp, 0);
  6631. if (entry)
  6632. entry->getName(str);
  6633. return str;
  6634. }
  6635. IStringVal& CLocalWUQuery::getQueryResTxtName(IStringVal &str) const
  6636. {
  6637. Owned<IConstWUAssociatedFile> entry = getAssociatedFile(FileTypeResText, 0);
  6638. if (entry)
  6639. entry->getName(str);
  6640. return str;
  6641. }
  6642. unsigned CLocalWUQuery::getQueryDllCrc() const
  6643. {
  6644. Owned<IConstWUAssociatedFile> entry = getAssociatedFile(FileTypeDll, 0);
  6645. if (entry)
  6646. return entry->getCrc();
  6647. return 0;
  6648. }
  6649. void CLocalWUQuery::setQueryText(const char *text)
  6650. {
  6651. p->setProp("Text", text);
  6652. }
  6653. void CLocalWUQuery::setQueryName(const char *qname)
  6654. {
  6655. p->setProp("@name", qname);
  6656. }
  6657. void CLocalWUQuery::setQueryMainDefinition(const char * str)
  6658. {
  6659. p->setProp("@main", str);
  6660. }
  6661. void CLocalWUQuery::addAssociatedFile(WUFileType type, const char * name, const char * ip, const char * desc, unsigned crc)
  6662. {
  6663. CriticalBlock block(crit);
  6664. loadAssociated();
  6665. if (!associated.length())
  6666. p->addPropTree("Associated", createPTree("Associated"));
  6667. IPropertyTree *pl = p->queryPropTree("Associated");
  6668. IPropertyTree *s = pl->addPropTree("File", createPTree("File"));
  6669. setEnum(s, "@type", type, queryFileTypes);
  6670. s->setProp("@filename", name);
  6671. s->setProp("@ip", ip);
  6672. s->setProp("@desc", desc);
  6673. if (crc)
  6674. s->setPropInt("@crc", crc);
  6675. IConstWUAssociatedFile * q = new CLocalWUAssociated(LINK(s));
  6676. associated.append(*q);
  6677. }
  6678. void CLocalWUQuery::removeAssociatedFile(WUFileType type, const char * name, const char * desc)
  6679. {
  6680. CriticalBlock block(crit);
  6681. associatedCached = false;
  6682. associated.kill();
  6683. StringBuffer xpath;
  6684. xpath.append("Associated/File");
  6685. if (type)
  6686. xpath.append("[@type=\"").append(getEnumText(type, queryFileTypes)).append("\"]");
  6687. if (name)
  6688. xpath.append("[@filename=\"").append(name).append("\"]");
  6689. if (desc)
  6690. xpath.append("[@desc=\"").append(desc).append("\"]");
  6691. p->removeProp(xpath.str());
  6692. }
  6693. void CLocalWUQuery::removeAssociatedFiles()
  6694. {
  6695. associatedCached = false;
  6696. associated.kill();
  6697. p->removeProp("Associated");
  6698. }
  6699. IConstWUAssociatedFile * CLocalWUQuery::getAssociatedFile(WUFileType type, unsigned index) const
  6700. {
  6701. CriticalBlock block(crit);
  6702. loadAssociated();
  6703. ForEachItemIn(idx, associated)
  6704. {
  6705. CLocalWUAssociated &cur = static_cast<CLocalWUAssociated &>(associated.item(idx));
  6706. if (cur.getType() == type)
  6707. {
  6708. if (index-- == 0)
  6709. return &OLINK(cur);
  6710. }
  6711. }
  6712. return NULL;
  6713. }
  6714. void CLocalWUQuery::addSpecialCaseAssociated(WUFileType type, const char * propname, unsigned crc) const
  6715. {
  6716. const char * name = p->queryProp(propname);
  6717. if (name)
  6718. {
  6719. IPropertyTree *s = createPTree("File");
  6720. setEnum(s, "@type", type, queryFileTypes);
  6721. s->setProp("@filename", name);
  6722. if (crc)
  6723. s->setPropInt("@crc", crc);
  6724. associated.append(*new CLocalWUAssociated(s));
  6725. }
  6726. }
  6727. void CLocalWUQuery::loadAssociated() const
  6728. {
  6729. CriticalBlock block(crit);
  6730. if (!associatedCached)
  6731. {
  6732. assertex(associated.length() == 0);
  6733. addSpecialCaseAssociated(FileTypeDll, "DllName", p->getPropInt("DllCrc", 0));
  6734. addSpecialCaseAssociated(FileTypeCpp, "CppName", 0);
  6735. addSpecialCaseAssociated(FileTypeResText, "ResTxtName", 0);
  6736. Owned<IPropertyTreeIterator> r = p->getElements("Associated/File");
  6737. for (r->first(); r->isValid(); r->next())
  6738. {
  6739. IPropertyTree *rp = &r->query();
  6740. rp->Link();
  6741. associated.append(*new CLocalWUAssociated(rp));
  6742. }
  6743. associatedCached = true;
  6744. }
  6745. }
  6746. IConstWUAssociatedFileIterator& CLocalWUQuery::getAssociatedFiles() const
  6747. {
  6748. CriticalBlock block(crit);
  6749. loadAssociated();
  6750. return *new CArrayIteratorOf<IConstWUAssociatedFile,IConstWUAssociatedFileIterator> (associated, 0, (IConstWUQuery *) this);
  6751. }
  6752. //========================================================================================
  6753. CLocalWUWebServicesInfo::CLocalWUWebServicesInfo(IPropertyTree *props) : p(props)
  6754. {
  6755. }
  6756. IStringVal& CLocalWUWebServicesInfo::getModuleName(IStringVal &str) const
  6757. {
  6758. str.set(p->queryProp("@module"));
  6759. return str;
  6760. }
  6761. IStringVal& CLocalWUWebServicesInfo::getAttributeName(IStringVal &str) const
  6762. {
  6763. str.set(p->queryProp("@attribute"));
  6764. return str;
  6765. }
  6766. IStringVal& CLocalWUWebServicesInfo::getDefaultName(IStringVal &str) const
  6767. {
  6768. str.set(p->queryProp("@defaultName"));
  6769. return str;
  6770. }
  6771. unsigned CLocalWUWebServicesInfo::getWebServicesCRC() const
  6772. {
  6773. return (unsigned) p->getPropInt("@crc");
  6774. }
  6775. IStringVal& CLocalWUWebServicesInfo::getInfo(const char *name, IStringVal &str) const
  6776. {
  6777. if (!name)
  6778. {
  6779. StringBuffer ws_info;
  6780. ws_info.appendf("<%s ", p->queryName());
  6781. Owned<IAttributeIterator> attrs = p->getAttributes();
  6782. for(attrs->first(); attrs->isValid(); attrs->next())
  6783. {
  6784. const char *name = attrs->queryName()+1;
  6785. const char *value = attrs->queryValue();
  6786. ws_info.appendf("%s='%s' ", name, value);
  6787. }
  6788. ws_info.append("> \n");
  6789. Owned<IPropertyTreeIterator> info = p->getElements("*");
  6790. ForEach(*info)
  6791. {
  6792. IPropertyTree &item = info->query();
  6793. const char *name = item.queryName();
  6794. if (name)
  6795. {
  6796. MemoryBuffer mb;
  6797. bool isbin = p->isBinary(name);
  6798. if (isbin)
  6799. {
  6800. p->getPropBin(name,mb);
  6801. if (mb.length())
  6802. {
  6803. unsigned len = 0;
  6804. mb.read(len);
  6805. StringBuffer encodedString;
  6806. StringBuffer val(len, (const char *) mb.readDirect(len));
  6807. encodeXML(val, encodedString);
  6808. ws_info.appendf("<%s>%s</%s>", name, encodedString.str(), name);
  6809. }
  6810. }
  6811. else
  6812. {
  6813. StringBuffer tmp;
  6814. toXML(&item, tmp);
  6815. ws_info.append(tmp.str());
  6816. }
  6817. }
  6818. }
  6819. ws_info.appendf("</%s>", p->queryName());
  6820. str.setLen(ws_info.str(), ws_info.length());
  6821. }
  6822. else
  6823. {
  6824. MemoryBuffer mb;
  6825. p->getPropBin(name,mb);
  6826. if (mb.length())
  6827. {
  6828. unsigned len;
  6829. mb.read(len);
  6830. str.setLen((const char *) mb.readDirect(len), len);
  6831. }
  6832. }
  6833. return str;
  6834. }
  6835. IStringVal& CLocalWUWebServicesInfo::getText(const char *name, IStringVal &str) const
  6836. {
  6837. str.set(p->queryProp(name));
  6838. return str;
  6839. }
  6840. void CLocalWUWebServicesInfo::setModuleName(const char *mname)
  6841. {
  6842. p->setProp("@module", mname);
  6843. }
  6844. void CLocalWUWebServicesInfo::setAttributeName(const char *aname)
  6845. {
  6846. p->setProp("@attribute", aname);
  6847. }
  6848. void CLocalWUWebServicesInfo::setDefaultName(const char *dname)
  6849. {
  6850. p->setProp("@defaultName", dname);
  6851. }
  6852. void CLocalWUWebServicesInfo::setWebServicesCRC(unsigned crc)
  6853. {
  6854. p->setPropInt("@crc", crc);
  6855. }
  6856. void CLocalWUWebServicesInfo::setInfo(const char *name, const char *info)
  6857. {
  6858. MemoryBuffer m;
  6859. unsigned len = (size32_t)strlen(info);
  6860. serializeLPString(len, info, m);
  6861. p->setPropBin(name, m.length(), m.toByteArray());
  6862. }
  6863. void CLocalWUWebServicesInfo::setText(const char *name, const char *info)
  6864. {
  6865. p->setProp(name, info);
  6866. }
  6867. //========================================================================================
  6868. CLocalWURoxieQueryInfo::CLocalWURoxieQueryInfo(IPropertyTree *props) : p(props)
  6869. {
  6870. }
  6871. IStringVal& CLocalWURoxieQueryInfo::getQueryInfo(IStringVal &str) const
  6872. {
  6873. IPropertyTree *queryTree = p->queryPropTree("query");
  6874. if (queryTree)
  6875. {
  6876. StringBuffer temp;
  6877. toXML(queryTree, temp);
  6878. str.set(temp.str());
  6879. }
  6880. return str;
  6881. }
  6882. IStringVal& CLocalWURoxieQueryInfo::getDefaultPackageInfo(IStringVal &str) const
  6883. {
  6884. MemoryBuffer mb;
  6885. p->getPropBin("RoxiePackages",mb);
  6886. if (mb.length())
  6887. {
  6888. unsigned len;
  6889. mb.read(len);
  6890. str.setLen((const char *) mb.readDirect(len), len);
  6891. }
  6892. return str;
  6893. }
  6894. IStringVal& CLocalWURoxieQueryInfo::getRoxieClusterName(IStringVal &str) const
  6895. {
  6896. const char *val = p->queryProp("@roxieClusterName");
  6897. if (val)
  6898. str.set(val);
  6899. return str;
  6900. }
  6901. IStringVal& CLocalWURoxieQueryInfo::getWuid(IStringVal &str) const
  6902. {
  6903. const char *val = p->queryProp("@wuid");
  6904. if (val)
  6905. str.set(val);
  6906. return str;
  6907. }
  6908. void CLocalWURoxieQueryInfo::setQueryInfo(const char *info)
  6909. {
  6910. IPropertyTree *queryTree = p->queryPropTree("query");
  6911. if (queryTree)
  6912. p->removeTree(queryTree);
  6913. IPropertyTree * tempTree = p->addPropTree("query", createPTreeFromXMLString(info));
  6914. if (!p->hasProp("@roxieClusterName"))
  6915. {
  6916. const char *roxieClusterName = tempTree->queryProp("@roxieName");
  6917. if (roxieClusterName && *roxieClusterName)
  6918. p->addProp("@roxieClusterName", roxieClusterName);
  6919. }
  6920. if (!p->hasProp("@wuid"))
  6921. {
  6922. const char *wuid = tempTree->queryProp("Query/@wuid");
  6923. if (wuid && *wuid)
  6924. p->addProp("@wuid", wuid);
  6925. }
  6926. }
  6927. void CLocalWURoxieQueryInfo::setDefaultPackageInfo(const char *info, int len)
  6928. {
  6929. MemoryBuffer m;
  6930. serializeLPString(len, info, m);
  6931. p->setPropBin("RoxiePackages", m.length(), m.toByteArray());
  6932. }
  6933. void CLocalWURoxieQueryInfo::setRoxieClusterName(const char *info)
  6934. {
  6935. p->setProp("@roxieClusterName", info);
  6936. }
  6937. void CLocalWURoxieQueryInfo::setWuid(const char *info)
  6938. {
  6939. p->setProp("@wuid", info);
  6940. }
  6941. //========================================================================================
  6942. CLocalWUResult::CLocalWUResult(IPropertyTree *props) : p(props)
  6943. {
  6944. }
  6945. mapEnums resultStatuses[] = {
  6946. { ResultStatusUndefined, "undefined" },
  6947. { ResultStatusCalculated, "calculated" },
  6948. { ResultStatusSupplied, "supplied" },
  6949. { ResultStatusFailed, "failed" },
  6950. { ResultStatusPartial, "partial" },
  6951. { ResultStatusSize, NULL }
  6952. };
  6953. WUResultStatus CLocalWUResult::getResultStatus() const
  6954. {
  6955. return (WUResultStatus ) getEnum(p, "@status", resultStatuses);
  6956. }
  6957. IStringVal& CLocalWUResult::getResultName(IStringVal &str) const
  6958. {
  6959. str.set(p->queryProp("@name"));
  6960. return str;
  6961. }
  6962. int CLocalWUResult::getResultSequence() const
  6963. {
  6964. return p->getPropInt("@sequence", -1);
  6965. }
  6966. bool CLocalWUResult::isResultScalar() const
  6967. {
  6968. return p->getPropInt("@isScalar", 1) != 0;
  6969. }
  6970. bool findSize(int size, IntArray &sizes)
  6971. {
  6972. ForEachItemIn(idx, sizes)
  6973. {
  6974. if (sizes.item(idx)==size)
  6975. return true;
  6976. }
  6977. return false;
  6978. }
  6979. void CLocalWUResult::getSchema(TypeInfoArray &types, StringAttrArray &names, IStringVal * eclText) const
  6980. {
  6981. MemoryBuffer schema;
  6982. p->getPropBin("SchemaRaw", schema);
  6983. if (schema.length())
  6984. {
  6985. for (;;)
  6986. {
  6987. StringAttr name;
  6988. schema.read(name);
  6989. if (*schema.readDirect(0)==type_void)
  6990. break;
  6991. names.append(*new StringAttrItem(name));
  6992. types.append(*deserializeType(schema)); // MORE - nested records!
  6993. }
  6994. schema.skip(1);
  6995. if (schema.length() != schema.getPos())
  6996. {
  6997. unsigned eclLen;
  6998. schema.read(eclLen);
  6999. const char * schemaData = (const char *)schema.readDirect(eclLen);
  7000. if (eclText)
  7001. {
  7002. eclText->setLen(schemaData, eclLen);
  7003. if ((eclLen == 0) && names.ordinality())
  7004. {
  7005. const char * firstName = names.item(0).text;
  7006. StringBuffer temp;
  7007. temp.append("RECORD ");
  7008. types.item(0).getECLType(temp);
  7009. temp.append(" value{NAMED('").append(firstName).append("')}").append("; END;");
  7010. eclText->set(temp.str());
  7011. }
  7012. }
  7013. }
  7014. }
  7015. }
  7016. void readRow(StringBuffer &out, MemoryBuffer &in, TypeInfoArray &types, StringAttrArray &names)
  7017. {
  7018. ForEachItemIn(idx, types)
  7019. {
  7020. StringAttrItem &name = names.item(idx);
  7021. ITypeInfo &type = types.item(idx);
  7022. unsigned size = type.getSize();
  7023. switch(type.getTypeCode())
  7024. {
  7025. case type_data:
  7026. if (size==UNKNOWN_LENGTH)
  7027. {
  7028. if (in.remaining() < sizeof(int))
  7029. throw MakeStringException(WUERR_CorruptResult, "corrupt workunit information");
  7030. in.read(size);
  7031. }
  7032. outputXmlData(size, in.readDirect(size), name.text, out);
  7033. break;
  7034. case type_string:
  7035. if (size==UNKNOWN_LENGTH)
  7036. {
  7037. if (in.remaining() < sizeof(int))
  7038. throw MakeStringException(WUERR_CorruptResult, "corrupt workunit information");
  7039. in.read(size);
  7040. }
  7041. outputXmlString(size, (const char *) in.readDirect(size), name.text, out);
  7042. break;
  7043. case type_varstring:
  7044. {
  7045. if (size == UNKNOWN_LENGTH)
  7046. size = (size32_t)strlen((const char *) in.readDirect(0))+1;
  7047. const char * text = (const char *) in.readDirect(size);
  7048. outputXmlString((size32_t)strlen(text), text, name.text, out);
  7049. break;
  7050. }
  7051. case type_unicode:
  7052. {
  7053. unsigned len = type.getStringLen();
  7054. if (size==UNKNOWN_LENGTH)
  7055. in.read(len);
  7056. outputXmlUnicode(len, (UChar const *) in.readDirect(len*2), name.text, out);
  7057. }
  7058. break;
  7059. case type_utf8:
  7060. {
  7061. unsigned len = type.getStringLen();
  7062. if (size==UNKNOWN_LENGTH)
  7063. {
  7064. in.read(len);
  7065. size = rtlUtf8Size(len, in.readDirect(0));
  7066. }
  7067. outputXmlUtf8(len, (const char *) in.readDirect(size), name.text, out);
  7068. }
  7069. break;
  7070. case type_qstring:
  7071. {
  7072. unsigned len = type.getStringLen();
  7073. if (size==UNKNOWN_LENGTH)
  7074. in.read(len);
  7075. unsigned outlen;
  7076. char *outstr;
  7077. rtlQStrToStrX(outlen, outstr, len, (const char *) in.readDirect(rtlQStrSize(len)));
  7078. outputXmlString(outlen, outstr, name.text, out);
  7079. free(outstr);
  7080. break;
  7081. }
  7082. case type_int:
  7083. case type_swapint:
  7084. if (type.isSigned())
  7085. {
  7086. const unsigned char *raw = (const unsigned char *) in.readDirect(size);
  7087. unsigned __int64 cval8 = 0;
  7088. //MORE: I think this is wrong - swapped doesn't mean little/big/
  7089. if (type.isSwappedEndian())
  7090. {
  7091. unsigned idx = 0;
  7092. if (raw[idx] & 0x80)
  7093. cval8 = (__int64)-1;
  7094. while (size--)
  7095. cval8 = (cval8 << 8) | raw[idx++];
  7096. }
  7097. else
  7098. {
  7099. if (raw[size-1] & 0x80)
  7100. cval8 = (__int64)-1;
  7101. while (size--)
  7102. cval8 = (cval8 << 8) | raw[size];
  7103. }
  7104. outputXmlInt((__int64) cval8, name.text, out);
  7105. }
  7106. else
  7107. {
  7108. const unsigned char *raw = (const unsigned char *) in.readDirect(size);
  7109. unsigned __int64 cval8 = 0;
  7110. if (type.isSwappedEndian())
  7111. {
  7112. unsigned idx = 0;
  7113. while (size--)
  7114. cval8 = (cval8 << 8) | raw[idx++];
  7115. }
  7116. else
  7117. {
  7118. while (size--)
  7119. cval8 = (cval8 << 8) | raw[size];
  7120. }
  7121. outputXmlUInt(cval8, name.text, out);
  7122. }
  7123. break;
  7124. case type_boolean:
  7125. bool cvalb;
  7126. in.read(cvalb);
  7127. outputXmlBool(cvalb, name.text, out);
  7128. break;
  7129. case type_decimal:
  7130. if (type.isSigned())
  7131. outputXmlDecimal(in.readDirect(size), size, type.getPrecision(), name.text, out);
  7132. else
  7133. outputXmlUDecimal(in.readDirect(size), size, type.getPrecision(), name.text, out);
  7134. break;
  7135. case type_real:
  7136. double cvald;
  7137. switch(size)
  7138. {
  7139. case 4:
  7140. float cvalf;
  7141. in.read(cvalf);
  7142. cvald = cvalf;
  7143. break;
  7144. case 8:
  7145. in.read(cvald);
  7146. break;
  7147. }
  7148. outputXmlReal(cvald, name.text, out);
  7149. break;
  7150. default:
  7151. assertex(!"unexpected type in raw record");
  7152. break;
  7153. }
  7154. }
  7155. }
  7156. IStringVal& CLocalWUResult::getResultXml(IStringVal &str, bool hidePassword) const
  7157. {
  7158. TypeInfoArray types;
  7159. StringAttrArray names;
  7160. getSchema(types, names);
  7161. StringBuffer xml;
  7162. const char * name = p->queryProp("@name");
  7163. if (name)
  7164. xml.appendf("<Dataset name=\'%s\'>\n", name);
  7165. else
  7166. xml.append("<Dataset>\n");
  7167. if (hidePassword && p->getPropBool("Format/@password"))
  7168. {
  7169. xml.append(" <Row>");
  7170. appendXMLTag(xml, name, "****");
  7171. xml.append("</Row>\n");
  7172. }
  7173. else if (p->hasProp("Value"))
  7174. {
  7175. MemoryBuffer raw;
  7176. p->getPropBin("Value", raw);
  7177. unsigned __int64 numrows = getResultRowCount();
  7178. while (numrows--)
  7179. {
  7180. xml.append(" <Row>");
  7181. readRow(xml, raw, types, names);
  7182. xml.append("</Row>\n");
  7183. }
  7184. }
  7185. else if (p->hasProp("xmlValue"))
  7186. {
  7187. xml.append(" <Row>");
  7188. appendXMLTag(xml, name, p->queryProp("xmlValue"));
  7189. xml.append("</Row>\n");
  7190. }
  7191. xml.append("</Dataset>\n");
  7192. str.set(xml.str());
  7193. return str;
  7194. }
  7195. IProperties *CLocalWUResult::queryResultXmlns()
  7196. {
  7197. CriticalBlock block(crit);
  7198. if (xmlns)
  7199. return xmlns;
  7200. xmlns.setown(createProperties());
  7201. Owned<IAttributeIterator> it = p->getAttributes();
  7202. unsigned prefixLen = strlen("@xmlns");
  7203. ForEach(*it)
  7204. {
  7205. const char *name = it->queryName();
  7206. if (!strncmp("@xmlns", name, prefixLen))
  7207. {
  7208. if (name[prefixLen]==':') //normal case
  7209. xmlns->setProp(name+prefixLen+1, it->queryValue());
  7210. else if (!name[prefixLen]) //special case, unprefixed namespace
  7211. xmlns->setProp("xmlns", it->queryValue());
  7212. }
  7213. }
  7214. return xmlns;
  7215. }
  7216. unsigned CLocalWUResult::getResultFetchSize() const
  7217. {
  7218. return p->getPropInt("fetchSize", 100);
  7219. }
  7220. __int64 CLocalWUResult::getResultTotalRowCount() const
  7221. {
  7222. return p->getPropInt64("totalRowCount", -1);
  7223. }
  7224. __int64 CLocalWUResult::getResultRowCount() const
  7225. {
  7226. return p->getPropInt64("rowCount", 0);
  7227. }
  7228. void CLocalWUResult::getResultDataset(IStringVal & ecl, IStringVal & defs) const
  7229. {
  7230. ecl.set(p->queryProp("datasetEcl"));
  7231. defs.set(p->queryProp("datasetEclDefs"));
  7232. }
  7233. IStringVal& CLocalWUResult::getResultLogicalName(IStringVal & val) const
  7234. {
  7235. val.set(p->queryProp("logicalName"));
  7236. return val;
  7237. }
  7238. IStringVal& CLocalWUResult::getResultKeyField(IStringVal & ecl) const
  7239. {
  7240. ecl.set(p->queryProp("keyField"));
  7241. return ecl;
  7242. }
  7243. unsigned CLocalWUResult::getResultRequestedRows() const
  7244. {
  7245. return p->getPropInt("requestedRows", 1);
  7246. }
  7247. IStringVal& CLocalWUResult::getResultEclSchema(IStringVal & str) const
  7248. {
  7249. TypeInfoArray types;
  7250. StringAttrArray names;
  7251. getSchema(types, names, &str);
  7252. return str;
  7253. }
  7254. IStringVal& CLocalWUResult::getResultRecordSizeEntry(IStringVal & str) const
  7255. {
  7256. str.set(p->queryProp("@recordSizeEntry"));
  7257. return str;
  7258. }
  7259. IStringVal& CLocalWUResult::getResultTransformerEntry(IStringVal & str) const
  7260. {
  7261. str.set(p->queryProp("@transformerEntry"));
  7262. return str;
  7263. }
  7264. __int64 CLocalWUResult::getResultRowLimit() const
  7265. {
  7266. return p->getPropInt64("@rowLimit");
  7267. }
  7268. IStringVal& CLocalWUResult::getResultFilename(IStringVal & str) const
  7269. {
  7270. str.set(p->queryProp("@tempFilename"));
  7271. return str;
  7272. }
  7273. IStringVal& CLocalWUResult::getResultFieldOpt(const char *name, IStringVal &str) const
  7274. {
  7275. str.clear();
  7276. if (!name || !*name)
  7277. return str;
  7278. IPropertyTree *format = p->queryPropTree("Format");
  7279. if (!format)
  7280. return str;
  7281. VStringBuffer xpath("@%s", name);
  7282. str.set(format->queryProp(xpath));
  7283. return str;
  7284. }
  7285. void CLocalWUResult::setResultStatus(WUResultStatus status)
  7286. {
  7287. setEnum(p, "@status", status, resultStatuses);
  7288. if (status==ResultStatusUndefined)
  7289. p->removeProp("Value");
  7290. }
  7291. void CLocalWUResult::setResultName(const char *s)
  7292. {
  7293. p->setProp("@name", s);
  7294. }
  7295. void CLocalWUResult::setResultSequence(unsigned seq)
  7296. {
  7297. p->setPropInt("@sequence", seq);
  7298. }
  7299. void CLocalWUResult::setResultSchemaRaw(unsigned size, const void *schema)
  7300. {
  7301. p->setPropBin("SchemaRaw", size, schema);
  7302. }
  7303. void CLocalWUResult::setResultXmlns(const char *prefix, const char *uri)
  7304. {
  7305. StringBuffer xpath("@xmlns");
  7306. if (prefix && *prefix)
  7307. xpath.append(':').append(prefix);
  7308. p->setProp(xpath, uri);
  7309. }
  7310. void CLocalWUResult::setResultFieldOpt(const char *name, const char *value)
  7311. {
  7312. if (!name || !*name)
  7313. return;
  7314. IPropertyTree *format = ensurePTree(p, "Format");
  7315. VStringBuffer xpath("@%s", name);
  7316. format->setProp(xpath, value);
  7317. }
  7318. void CLocalWUResult::setResultScalar(bool isScalar)
  7319. {
  7320. p->setPropInt("@isScalar", (int) isScalar);
  7321. if (isScalar)
  7322. setResultTotalRowCount(1);
  7323. }
  7324. void CLocalWUResult::setResultRaw(unsigned len, const void *data, WUResultFormat format)
  7325. {
  7326. p->setPropBin("Value", len, data);
  7327. setResultStatus(ResultStatusSupplied);
  7328. setResultFormat(format);
  7329. }
  7330. void CLocalWUResult::setResultFormat(WUResultFormat format)
  7331. {
  7332. switch (format)
  7333. {
  7334. case ResultFormatXml:
  7335. p->setProp("@format","xml");
  7336. break;
  7337. case ResultFormatXmlSet:
  7338. p->setProp("@format","xmlSet");
  7339. break;
  7340. case ResultFormatCsv:
  7341. p->setProp("@format","csv");
  7342. break;
  7343. default:
  7344. p->removeProp("@format");
  7345. break;
  7346. }
  7347. }
  7348. void CLocalWUResult::setResultXML(const char *val)
  7349. {
  7350. p->setProp("xmlValue", val);
  7351. }
  7352. void CLocalWUResult::addResultRaw(unsigned len, const void *data, WUResultFormat format)
  7353. {
  7354. p->appendPropBin("Value", len, data);
  7355. setResultStatus(ResultStatusPartial);
  7356. const char *existingFormat = p->queryProp("@format");
  7357. const char *formatStr = NULL;
  7358. switch (format)
  7359. {
  7360. case ResultFormatXml:
  7361. formatStr = "xml";
  7362. break;
  7363. case ResultFormatXmlSet:
  7364. formatStr = "xmlSet";
  7365. break;
  7366. case ResultFormatCsv:
  7367. formatStr = "csv";
  7368. break;
  7369. default:
  7370. p->removeProp("@format");
  7371. break;
  7372. }
  7373. if (format)
  7374. {
  7375. if (existingFormat)
  7376. {
  7377. if (0 != stricmp(formatStr, existingFormat))
  7378. throw MakeStringException(WUERR_ResultFormatMismatch, "addResult format %s, does not match existing format %s", formatStr, existingFormat);
  7379. }
  7380. else
  7381. p->setProp("@format", formatStr);
  7382. }
  7383. }
  7384. void CLocalWUResult::setResultFetchSize(unsigned rows)
  7385. {
  7386. p->setPropInt("fetchSize", rows);
  7387. }
  7388. void CLocalWUResult::setResultTotalRowCount(__int64 rows)
  7389. {
  7390. p->setPropInt64("totalRowCount", rows);
  7391. }
  7392. void CLocalWUResult::setResultRowCount(__int64 rows)
  7393. {
  7394. p->setPropInt64("rowCount", rows);
  7395. }
  7396. void CLocalWUResult::setResultDataset(const char *ecl, const char *defs)
  7397. {
  7398. p->setProp("datasetEcl", ecl);
  7399. p->setProp("datasetEclDefs", defs);
  7400. }
  7401. void CLocalWUResult::setResultLogicalName(const char *logicalName)
  7402. {
  7403. p->setProp("logicalName", logicalName);
  7404. }
  7405. void CLocalWUResult::setResultKeyField(const char *ecl)
  7406. {
  7407. p->setProp("keyField", ecl);
  7408. }
  7409. void CLocalWUResult::setResultRequestedRows(unsigned rows)
  7410. {
  7411. p->setPropInt("requestedRows", rows);
  7412. }
  7413. void CLocalWUResult::setResultRecordSizeEntry(const char * entry)
  7414. {
  7415. p->setProp("@recordSizeEntry", entry);
  7416. }
  7417. void CLocalWUResult::setResultTransformerEntry(const char * entry)
  7418. {
  7419. p->setProp("@transformerEntry", entry);
  7420. }
  7421. void CLocalWUResult::setResultRowLimit(__int64 value)
  7422. {
  7423. p->setPropInt64("@rowLimit", value);
  7424. }
  7425. void CLocalWUResult::setResultFilename(const char * name)
  7426. {
  7427. p->setProp("@tempFilename", name);
  7428. }
  7429. // MORE - it's an undetected error if we call getResult... of a type that does not match schema
  7430. __int64 CLocalWUResult::getResultInt() const
  7431. {
  7432. __int64 result = 0;
  7433. MemoryBuffer s;
  7434. p->getPropBin("Value", s);
  7435. if (s.length())
  7436. s.read(result);
  7437. else
  7438. result = p->getPropInt64("xmlValue");
  7439. return result;
  7440. }
  7441. bool CLocalWUResult::getResultBool() const
  7442. {
  7443. bool result = false;
  7444. MemoryBuffer s;
  7445. p->getPropBin("Value", s);
  7446. if (s.length())
  7447. s.read(result);
  7448. else
  7449. result = p->getPropBool("xmlValue");
  7450. return result;
  7451. }
  7452. double CLocalWUResult::getResultReal() const
  7453. {
  7454. double result = 0;
  7455. MemoryBuffer s;
  7456. p->getPropBin("Value", s);
  7457. if (s.length())
  7458. s.read(result);
  7459. else
  7460. {
  7461. const char *xmlVal = p->queryProp("xmlValue");
  7462. if (xmlVal)
  7463. result = atof(xmlVal);
  7464. }
  7465. return result;
  7466. }
  7467. void CLocalWUResult::getResultDecimal(void * val, unsigned len, unsigned precision, bool isSigned) const
  7468. {
  7469. MemoryBuffer s;
  7470. p->getPropBin("Value", s);
  7471. if (s.length())
  7472. {
  7473. assertex(s.length() == len);
  7474. s.read(len, val);
  7475. }
  7476. else
  7477. {
  7478. const char *xmlVal = p->queryProp("xmlValue");
  7479. if (xmlVal)
  7480. {
  7481. Decimal d;
  7482. d.setString(strlen(xmlVal), xmlVal);
  7483. if (isSigned)
  7484. d.getDecimal(len, precision, val);
  7485. else
  7486. d.getUDecimal(len, precision, val);
  7487. }
  7488. else
  7489. memset(val, 0, len);
  7490. }
  7491. }
  7492. IStringVal& CLocalWUResult::getResultString(IStringVal & str, bool hidePassword) const
  7493. {
  7494. if (hidePassword && p->getPropBool("@password"))
  7495. {
  7496. str.set("****");
  7497. return str;
  7498. }
  7499. MemoryBuffer s;
  7500. p->getPropBin("Value", s);
  7501. if (s.length())
  7502. {
  7503. unsigned len;
  7504. s.read(len);
  7505. str.setLen((const char *) s.readDirect(len), len);
  7506. }
  7507. else
  7508. {
  7509. p->getPropBin("xmlValue", s);
  7510. if (p->isBinary("xmlValue"))
  7511. str.setLen(s.toByteArray(), s.length());
  7512. else
  7513. {
  7514. char *ascii = rtlUtf8ToVStr(rtlUtf8Length(s.length(), s.toByteArray()), s.toByteArray());
  7515. str.set(ascii);
  7516. rtlFree(ascii);
  7517. }
  7518. }
  7519. return str;
  7520. }
  7521. WUResultFormat CLocalWUResult::getResultFormat() const
  7522. {
  7523. const char * format = p->queryProp("@format");
  7524. if (!format)
  7525. return ResultFormatRaw;
  7526. else if (strcmp(format, "xml") == 0)
  7527. return ResultFormatXml;
  7528. else if (strcmp(format, "xmlSet") == 0)
  7529. return ResultFormatXmlSet;
  7530. else if (strcmp(format, "csv") == 0)
  7531. return ResultFormatCsv;
  7532. else
  7533. throw MakeStringException(WUERR_InvalidResultFormat, "Unrecognised result format %s", format);
  7534. }
  7535. IDataVal& CLocalWUResult::getResultRaw(IDataVal & data, IXmlToRawTransformer * xmlTransformer, ICsvToRawTransformer * csvTransformer) const
  7536. {
  7537. MemoryBuffer s;
  7538. p->getPropBin("Value", s);
  7539. unsigned len = s.length();
  7540. if (len)
  7541. {
  7542. WUResultFormat format = getResultFormat();
  7543. if (format == ResultFormatXml || format == ResultFormatXmlSet)
  7544. {
  7545. if (!xmlTransformer)
  7546. throw MakeStringException(WUERR_MissingFormatTranslator, "No transformer supplied to translate XML format result");
  7547. xmlTransformer->transform(data, len, s.readDirect(len), format == ResultFormatXml);
  7548. }
  7549. else if (format == ResultFormatCsv)
  7550. {
  7551. if (!csvTransformer)
  7552. throw MakeStringException(WUERR_MissingFormatTranslator, "No transformer supplied to translate Csv format result");
  7553. csvTransformer->transform(data, len, s.readDirect(len), true);
  7554. }
  7555. else
  7556. data.setLen(s.readDirect(len), len);
  7557. }
  7558. else
  7559. data.clear();
  7560. return data;
  7561. }
  7562. unsigned CLocalWUResult::getResultHash() const
  7563. {
  7564. MemoryBuffer s;
  7565. p->getPropBin("Value", s);
  7566. unsigned len = s.length();
  7567. const byte * data = (const byte *)s.toByteArray();
  7568. return ~hashc(data, len, ~0);
  7569. }
  7570. IDataVal& CLocalWUResult::getResultUnicode(IDataVal & data) const
  7571. {
  7572. MemoryBuffer s;
  7573. p->getPropBin("Value", s);
  7574. if (s.length())
  7575. {
  7576. unsigned len;
  7577. s.read(len);
  7578. data.setLen(s.readDirect(len*2), len*2);
  7579. }
  7580. else
  7581. {
  7582. StringBuffer utf8;
  7583. if (p->getProp("xmlValue", utf8))
  7584. {
  7585. unsigned outlen;
  7586. UChar *out;
  7587. rtlUtf8ToUnicodeX(outlen, out, utf8.length(), utf8.str());
  7588. data.setLen(out, outlen*2);
  7589. rtlFree(out);
  7590. }
  7591. else
  7592. data.clear();
  7593. }
  7594. return data;
  7595. }
  7596. __int64 CLocalWUResult::getResultRawSize(IXmlToRawTransformer * xmlTransformer, ICsvToRawTransformer * csvTransformer) const
  7597. {
  7598. WUResultFormat format = getResultFormat();
  7599. if (format == ResultFormatRaw)
  7600. {
  7601. //MORE: This should not load the whole property...
  7602. MemoryBuffer s;
  7603. p->getPropBin("Value", s);
  7604. return s.length();
  7605. }
  7606. else
  7607. {
  7608. MemoryBuffer temp;
  7609. MemoryBuffer2IDataVal adaptor(temp);
  7610. getResultRaw(adaptor, xmlTransformer, csvTransformer);
  7611. return temp.length();
  7612. }
  7613. }
  7614. IDataVal& CLocalWUResult::getResultRaw(IDataVal & data, __int64 from, __int64 length, IXmlToRawTransformer * xmlTransformer, ICsvToRawTransformer * csvTransformer) const
  7615. {
  7616. WUResultFormat format = getResultFormat();
  7617. if (format != ResultFormatRaw)
  7618. {
  7619. MemoryBuffer temp;
  7620. MemoryBuffer2IDataVal adaptor(temp);
  7621. getResultRaw(adaptor, xmlTransformer, csvTransformer);
  7622. unsigned len = temp.length();
  7623. if (from > len) from = len;
  7624. if (from + length > len) length = len - from;
  7625. data.setLen(temp.readDirect(len) + from, (size32_t)length);
  7626. return data;
  7627. }
  7628. else
  7629. {
  7630. //MORE: This should not load the whole property, and should be different from the code above...
  7631. MemoryBuffer s;
  7632. p->getPropBin("Value", s);
  7633. unsigned len = s.length();
  7634. if (from > len) from = len;
  7635. if (from + length > len) length = len - from;
  7636. data.setLen(s.readDirect(len) + from, (size32_t)length);
  7637. return data;
  7638. }
  7639. }
  7640. bool CLocalWUResult::getResultIsAll() const
  7641. {
  7642. return p->getPropBool("@isAll", false);
  7643. }
  7644. // MORE - it's an undetected error if we call setResult... of a type that does not match schema
  7645. void CLocalWUResult::setResultInt(__int64 val)
  7646. {
  7647. // Note: we always serialize scalar integer results as int8, and schema must reflect this
  7648. MemoryBuffer m;
  7649. serializeInt8(val, m);
  7650. p->setPropBin("Value", m.length(), m.toByteArray());
  7651. setResultRowCount(1);
  7652. setResultTotalRowCount(1);
  7653. }
  7654. void CLocalWUResult::setResultUInt(unsigned __int64 val)
  7655. {
  7656. setResultInt((__int64) val);
  7657. }
  7658. void CLocalWUResult::setResultReal(double val)
  7659. {
  7660. // Note: we always serialize scalar real results as real8, and schema must reflect this
  7661. MemoryBuffer m;
  7662. serializeReal8(val, m);
  7663. p->setPropBin("Value", m.length(), m.toByteArray());
  7664. setResultRowCount(1);
  7665. setResultTotalRowCount(1);
  7666. }
  7667. void CLocalWUResult::setResultBool(bool val)
  7668. {
  7669. MemoryBuffer m;
  7670. serializeBool(val, m);
  7671. p->setPropBin("Value", m.length(), m.toByteArray());
  7672. setResultRowCount(1);
  7673. setResultTotalRowCount(1);
  7674. }
  7675. void CLocalWUResult::setResultString(const char *val, unsigned len)
  7676. {
  7677. // Note: we always serialize scalar strings with length prefix, and schema must reflect this
  7678. MemoryBuffer m;
  7679. serializeLPString(len, val, m);
  7680. p->setPropBin("Value", m.length(), m.toByteArray());
  7681. setResultRowCount(1);
  7682. setResultTotalRowCount(1);
  7683. }
  7684. void CLocalWUResult::setResultUnicode(const void *val, unsigned len)
  7685. {
  7686. // Note: we always serialize scalar strings with length prefix, and schema must reflect this
  7687. MemoryBuffer m;
  7688. m.append(len).append(len*2, val);
  7689. p->setPropBin("Value", m.length(), m.toByteArray());
  7690. setResultRowCount(1);
  7691. setResultTotalRowCount(1);
  7692. }
  7693. void CLocalWUResult::setResultData(const void *val, unsigned len)
  7694. {
  7695. // Note: we always serialize scalar data with length prefix, and schema must reflect this
  7696. MemoryBuffer m;
  7697. serializeLPString(len, (const char *)val, m);
  7698. p->setPropBin("Value", m.length(), m.toByteArray());
  7699. setResultRowCount(1);
  7700. setResultTotalRowCount(1);
  7701. }
  7702. void CLocalWUResult::setResultDecimal(const void *val, unsigned len)
  7703. {
  7704. // Note: serialized as data but with length known from schema
  7705. MemoryBuffer m;
  7706. serializeFixedData(len, val, m);
  7707. p->setPropBin("Value", m.length(), m.toByteArray());
  7708. setResultRowCount(1);
  7709. setResultTotalRowCount(1);
  7710. }
  7711. void CLocalWUResult::setResultRow(unsigned len, const void * data)
  7712. {
  7713. p->setPropBin("Value", len, data);
  7714. setResultRowCount(1);
  7715. setResultTotalRowCount(1);
  7716. setResultFormat(ResultFormatRaw);
  7717. }
  7718. void CLocalWUResult::setResultIsAll(bool value)
  7719. {
  7720. p->setPropBool("@isAll", value);
  7721. }
  7722. //==========================================================================================
  7723. CLocalWUPlugin::CLocalWUPlugin(IPropertyTree *props) : p(props)
  7724. {
  7725. }
  7726. IStringVal& CLocalWUPlugin::getPluginName(IStringVal &str) const
  7727. {
  7728. str.set(p->queryProp("@dllname"));
  7729. return str;
  7730. }
  7731. IStringVal& CLocalWUPlugin::getPluginVersion(IStringVal &str) const
  7732. {
  7733. str.set(p->queryProp("@version"));
  7734. return str;
  7735. }
  7736. void CLocalWUPlugin::setPluginName(const char *str)
  7737. {
  7738. p->setProp("@dllname", str);
  7739. }
  7740. void CLocalWUPlugin::setPluginVersion(const char *str)
  7741. {
  7742. p->setProp("@version", str);
  7743. }
  7744. //==========================================================================================
  7745. CLocalWULibrary::CLocalWULibrary(IPropertyTree *props) : p(props)
  7746. {
  7747. }
  7748. IStringVal& CLocalWULibrary::getName(IStringVal &str) const
  7749. {
  7750. str.set(p->queryProp("@name"));
  7751. return str;
  7752. }
  7753. void CLocalWULibrary::setName(const char *str)
  7754. {
  7755. p->setProp("@name", str);
  7756. }
  7757. //==========================================================================================
  7758. CLocalWUException::CLocalWUException(IPropertyTree *props) : p(props)
  7759. {
  7760. }
  7761. IStringVal& CLocalWUException::getExceptionSource(IStringVal &str) const
  7762. {
  7763. str.set(p->queryProp("@source"));
  7764. return str;
  7765. }
  7766. IStringVal& CLocalWUException::getExceptionMessage(IStringVal &str) const
  7767. {
  7768. str.set(p->queryProp(NULL));
  7769. return str;
  7770. }
  7771. unsigned CLocalWUException::getExceptionCode() const
  7772. {
  7773. return p->getPropInt("@code", 0);
  7774. }
  7775. ErrorSeverity CLocalWUException::getSeverity() const
  7776. {
  7777. return (ErrorSeverity)p->getPropInt("@severity", SeverityError);
  7778. }
  7779. IStringVal & CLocalWUException::getTimeStamp(IStringVal & dt) const
  7780. {
  7781. dt.set(p->queryProp("@time"));
  7782. return dt;
  7783. }
  7784. IStringVal & CLocalWUException::getExceptionFileName(IStringVal & str) const
  7785. {
  7786. str.set(p->queryProp("@filename"));
  7787. return str;
  7788. }
  7789. unsigned CLocalWUException::getExceptionLineNo() const
  7790. {
  7791. return p->getPropInt("@row", 0);
  7792. }
  7793. unsigned CLocalWUException::getExceptionColumn() const
  7794. {
  7795. return p->getPropInt("@col", 0);
  7796. }
  7797. unsigned CLocalWUException::getSequence() const
  7798. {
  7799. return p->getPropInt("@sequence", 0);
  7800. }
  7801. void CLocalWUException::setExceptionSource(const char *str)
  7802. {
  7803. p->setProp("@source", str);
  7804. }
  7805. void CLocalWUException::setExceptionMessage(const char *str)
  7806. {
  7807. p->setProp(NULL, str);
  7808. }
  7809. void CLocalWUException::setExceptionCode(unsigned code)
  7810. {
  7811. p->setPropInt("@code", code);
  7812. }
  7813. void CLocalWUException::setSeverity(ErrorSeverity level)
  7814. {
  7815. p->setPropInt("@severity", level);
  7816. }
  7817. void CLocalWUException::setTimeStamp(const char *str)
  7818. {
  7819. p->setProp("@time", str);
  7820. }
  7821. void CLocalWUException::setExceptionFileName(const char *str)
  7822. {
  7823. p->setProp("@filename", str);
  7824. }
  7825. void CLocalWUException::setExceptionLineNo(unsigned r)
  7826. {
  7827. p->setPropInt("@row", r);
  7828. }
  7829. void CLocalWUException::setExceptionColumn(unsigned c)
  7830. {
  7831. p->setPropInt("@col", c);
  7832. }
  7833. //==========================================================================================
  7834. CLocalWUAppValue::CLocalWUAppValue(IPropertyTree *props,unsigned child): p(props)
  7835. {
  7836. StringAttrBuilder propPath(prop);
  7837. propPath.append("*[").append(child).append("]");
  7838. }
  7839. IStringVal & CLocalWUAppValue::getApplication(IStringVal & str) const
  7840. {
  7841. str.set(p->queryName());
  7842. return str;
  7843. }
  7844. IStringVal & CLocalWUAppValue::getName(IStringVal & str) const
  7845. {
  7846. IPropertyTree* val=p->queryPropTree(prop.str());
  7847. if(val)
  7848. str.set(val->queryName());
  7849. return str;
  7850. }
  7851. IStringVal & CLocalWUAppValue::getValue(IStringVal & str) const
  7852. {
  7853. str.set(p->queryProp(prop.str()));
  7854. return str;
  7855. }
  7856. //==========================================================================================
  7857. CLocalWUStatistic::CLocalWUStatistic(IPropertyTree *props) : p(props)
  7858. {
  7859. }
  7860. IStringVal & CLocalWUStatistic::getCreator(IStringVal & str) const
  7861. {
  7862. const char * creator = p->queryProp("@creator");
  7863. str.set(creator);
  7864. return str;
  7865. }
  7866. IStringVal & CLocalWUStatistic::getDescription(IStringVal & str, bool createDefault) const
  7867. {
  7868. const char * desc = p->queryProp("@desc");
  7869. if (desc)
  7870. {
  7871. str.set(desc); // legacy and in case it is overridden
  7872. }
  7873. else if (createDefault)
  7874. {
  7875. StatisticKind kind = getKind();
  7876. assertex(kind != StKindNone);
  7877. const char * scope = p->queryProp("@scope");
  7878. assertex(scope);
  7879. //Clean up the format of the scope when converting it to a description
  7880. StringBuffer descriptionText;
  7881. if (streq(scope, GLOBAL_SCOPE))
  7882. {
  7883. const char * creator = p->queryProp("@creator");
  7884. descriptionText.append(creator).append(":");
  7885. queryLongStatisticName(descriptionText, kind);
  7886. }
  7887. else
  7888. {
  7889. loop
  7890. {
  7891. char c = *scope++;
  7892. if (!c)
  7893. break;
  7894. if (c == ':')
  7895. descriptionText.append(": ");
  7896. else
  7897. descriptionText.append(c);
  7898. }
  7899. if (kind != StTimeElapsed)
  7900. queryLongStatisticName(descriptionText.append(": "), kind);
  7901. }
  7902. str.set(descriptionText);
  7903. }
  7904. else
  7905. str.clear();
  7906. return str;
  7907. }
  7908. IStringVal & CLocalWUStatistic::getType(IStringVal & str) const
  7909. {
  7910. StatisticKind kind = getKind();
  7911. if (kind != StKindNone)
  7912. str.set(queryStatisticName(kind));
  7913. return str;
  7914. }
  7915. IStringVal & CLocalWUStatistic::getFormattedValue(IStringVal & str) const
  7916. {
  7917. StringBuffer formatted;
  7918. formatStatistic(formatted, getValue(), getMeasure());
  7919. str.set(formatted);
  7920. return str;
  7921. }
  7922. StatisticCreatorType CLocalWUStatistic::getCreatorType() const
  7923. {
  7924. return queryCreatorType(p->queryProp("@c"));
  7925. }
  7926. StatisticScopeType CLocalWUStatistic::getScopeType() const
  7927. {
  7928. return queryScopeType(p->queryProp("@s"));
  7929. }
  7930. StatisticKind CLocalWUStatistic::getKind() const
  7931. {
  7932. return queryStatisticKind(p->queryProp("@kind"));
  7933. }
  7934. IStringVal & CLocalWUStatistic::getScope(IStringVal & str) const
  7935. {
  7936. const char * scope = p->queryProp("@scope");
  7937. str.set(scope);
  7938. return str;
  7939. }
  7940. StatisticMeasure CLocalWUStatistic::getMeasure() const
  7941. {
  7942. return queryMeasure(p->queryProp("@unit"));
  7943. }
  7944. unsigned __int64 CLocalWUStatistic::getValue() const
  7945. {
  7946. return p->getPropInt64("@value", 0);
  7947. }
  7948. unsigned __int64 CLocalWUStatistic::getCount() const
  7949. {
  7950. return p->getPropInt64("@count", 0);
  7951. }
  7952. unsigned __int64 CLocalWUStatistic::getMax() const
  7953. {
  7954. return p->getPropInt64("@max", 0);
  7955. }
  7956. unsigned __int64 CLocalWUStatistic::getTimestamp() const
  7957. {
  7958. return p->getPropInt64("@ts", 0);
  7959. }
  7960. bool CLocalWUStatistic::matches(const IStatisticsFilter * filter) const
  7961. {
  7962. if (!filter)
  7963. return true;
  7964. const char * creator = p->queryProp("@creator");
  7965. const char * scope = p->queryProp("@scope");
  7966. return filter->matches(getCreatorType(), creator, getScopeType(), scope, getMeasure(), getKind());
  7967. }
  7968. //==========================================================================================
  7969. CLocalWULegacyTiming::CLocalWULegacyTiming(IPropertyTree *props) : p(props)
  7970. {
  7971. }
  7972. IStringVal & CLocalWULegacyTiming::getCreator(IStringVal & str) const
  7973. {
  7974. str.clear();
  7975. return str;
  7976. }
  7977. IStringVal & CLocalWULegacyTiming::getDescription(IStringVal & str, bool createDefault) const
  7978. {
  7979. str.set(p->queryProp("@name"));
  7980. return str;
  7981. }
  7982. IStringVal & CLocalWULegacyTiming::getType(IStringVal & str) const
  7983. {
  7984. str.set(queryStatisticName(StTimeElapsed));
  7985. return str;
  7986. }
  7987. IStringVal & CLocalWULegacyTiming::getFormattedValue(IStringVal & str) const
  7988. {
  7989. StringBuffer formatted;
  7990. formatStatistic(formatted, getValue(), getMeasure());
  7991. str.set(formatted);
  7992. return str;
  7993. }
  7994. StatisticCreatorType CLocalWULegacyTiming::getCreatorType() const
  7995. {
  7996. return SCTunknown;
  7997. }
  7998. StatisticScopeType CLocalWULegacyTiming::getScopeType() const
  7999. {
  8000. return SSTnone;
  8001. }
  8002. StatisticKind CLocalWULegacyTiming::getKind() const
  8003. {
  8004. return StTimeElapsed;
  8005. }
  8006. IStringVal & CLocalWULegacyTiming::getScope(IStringVal & str) const
  8007. {
  8008. str.clear();
  8009. return str;
  8010. }
  8011. StatisticMeasure CLocalWULegacyTiming::getMeasure() const
  8012. {
  8013. return SMeasureTimeNs;
  8014. }
  8015. unsigned __int64 CLocalWULegacyTiming::getValue() const
  8016. {
  8017. return p->getPropInt64("@duration", 0) * 1000000;
  8018. }
  8019. unsigned __int64 CLocalWULegacyTiming::getCount() const
  8020. {
  8021. return p->getPropInt64("@count", 0);
  8022. }
  8023. unsigned __int64 CLocalWULegacyTiming::getMax() const
  8024. {
  8025. return p->getPropInt64("@max", 0);
  8026. }
  8027. unsigned __int64 CLocalWULegacyTiming::getTimestamp() const
  8028. {
  8029. return 0;
  8030. }
  8031. bool CLocalWULegacyTiming::matches(const IStatisticsFilter * filter) const
  8032. {
  8033. if (!filter)
  8034. return true;
  8035. const char * creator = p->queryProp("@creator");
  8036. const char * scope = p->queryProp("@scope");
  8037. return filter->matches(SCTall, NULL, SSTall, NULL, getMeasure(), getKind());
  8038. }
  8039. //==========================================================================================
  8040. extern WORKUNIT_API ILocalWorkUnit * createLocalWorkUnit(const char *xml)
  8041. {
  8042. Owned<CLocalWorkUnit> cw = new CLocalWorkUnit((ISecManager *) NULL, NULL);
  8043. if (xml)
  8044. cw->loadPTree(createPTreeFromXMLString(xml));
  8045. else
  8046. {
  8047. Owned<IPropertyTree> p = createPTree("W_LOCAL");
  8048. p->setProp("@xmlns:xsi", "http://www.w3.org/1999/XMLSchema-instance");
  8049. cw->loadPTree(p.getClear());
  8050. }
  8051. ILocalWorkUnit* ret = QUERYINTERFACE(&cw->lockRemote(false), ILocalWorkUnit);
  8052. return ret;
  8053. }
  8054. void exportWorkUnitToXMLWithHiddenPasswords(IPropertyTree *p, IIOStream &out, unsigned extraXmlFlags)
  8055. {
  8056. const char *name = p->queryName();
  8057. if (!name)
  8058. name = "__unnamed__";
  8059. StringBuffer temp;
  8060. writeStringToStream(out, appendPTreeOpenTag(temp, p, name, 1));
  8061. Owned<IPropertyTreeIterator> elems = p->getElements("*", iptiter_sort);
  8062. ForEach(*elems)
  8063. {
  8064. IPropertyTree &elem = elems->query();
  8065. if (streq(elem.queryName(), "Parameters"))
  8066. {
  8067. writeStringToStream(out, appendPTreeOpenTag(temp.clear().append(' '), &elem, "Parameters", 2).append('\n'));
  8068. Owned<IPropertyTreeIterator> params = elem.getElements("*", iptiter_sort);
  8069. ForEach(*params)
  8070. {
  8071. IPropertyTree &param = params->query();
  8072. const char *paramname = param.queryName();
  8073. VStringBuffer xpath("Variables/Variable[@name='%s']/Format/@password", paramname);
  8074. if (p->getPropBool(xpath))
  8075. writeStringToStream(out, appendXMLTag(temp.clear().append(" "), paramname, "****").append('\n'));
  8076. else
  8077. {
  8078. toXML(&param, out, 2, XML_Format|XML_SortTags|extraXmlFlags);
  8079. }
  8080. }
  8081. writeStringToStream(out, appendXMLCloseTag(temp.clear().append(' '), "Parameters").append('\n'));
  8082. }
  8083. else if (streq(elem.queryName(), "Variables"))
  8084. {
  8085. writeStringToStream(out, appendPTreeOpenTag(temp.clear().append(' '), &elem, "Variables", 2).append('\n'));
  8086. Owned<IPropertyTreeIterator> vars = elem.getElements("*", iptiter_sort);
  8087. ForEach(*vars)
  8088. {
  8089. Owned<IPropertyTree> var = LINK(&vars->query());
  8090. if (var->getPropBool("Format/@password"))
  8091. {
  8092. var.setown(createPTreeFromIPT(var)); //copy and remove password values
  8093. var->removeProp("Value");
  8094. var->removeProp("xmlValue");
  8095. }
  8096. toXML(var, out, 2, XML_Format|XML_SortTags|extraXmlFlags);
  8097. }
  8098. writeStringToStream(out, appendXMLCloseTag(temp.clear().append(' '), "Variables").append('\n'));
  8099. }
  8100. else
  8101. toXML(&elem, out, 1, XML_Format|XML_SortTags|extraXmlFlags);
  8102. }
  8103. writeStringToStream(out, appendXMLCloseTag(temp.clear(), name));
  8104. }
  8105. StringBuffer &exportWorkUnitToXMLWithHiddenPasswords(IPropertyTree *p, StringBuffer &str)
  8106. {
  8107. class CAdapter : public CInterface, implements IIOStream
  8108. {
  8109. StringBuffer &out;
  8110. public:
  8111. IMPLEMENT_IINTERFACE;
  8112. CAdapter(StringBuffer &_out) : out(_out) { }
  8113. virtual void flush() { }
  8114. virtual size32_t read(size32_t len, void * data) { UNIMPLEMENTED; return 0; }
  8115. virtual size32_t write(size32_t len, const void * data) { out.append(len, (const char *)data); return len; }
  8116. } adapter(str);
  8117. exportWorkUnitToXMLWithHiddenPasswords(p->queryBranch(NULL), adapter, 0);
  8118. return str;
  8119. }
  8120. void exportWorkUnitToXMLFileWithHiddenPasswords(IPropertyTree *p, const char *filename, unsigned extraXmlFlags)
  8121. {
  8122. OwnedIFile ifile = createIFile(filename);
  8123. OwnedIFileIO ifileio = ifile->open(IFOcreate);
  8124. Owned<IIOStream> stream = createIOStream(ifileio);
  8125. exportWorkUnitToXMLWithHiddenPasswords(p->queryBranch(NULL), *stream, extraXmlFlags);
  8126. }
  8127. extern WORKUNIT_API StringBuffer &exportWorkUnitToXML(const IConstWorkUnit *wu, StringBuffer &str, bool unpack, bool includeProgress, bool hidePasswords)
  8128. {
  8129. // MORE - queryPTree isn't really safe without holding CLocalWorkUnit::crit - really need to move these functions into CLocalWorkunit
  8130. const IExtendedWUInterface *ewu = queryExtendedWU(wu);
  8131. if (ewu)
  8132. {
  8133. Linked<IPropertyTree> p;
  8134. if (unpack||includeProgress)
  8135. p.setown(ewu->getUnpackedTree(includeProgress));
  8136. else
  8137. p.set(ewu->queryPTree());
  8138. if (hidePasswords && p->hasProp("Variables/Variable[Format/@password]"))
  8139. return exportWorkUnitToXMLWithHiddenPasswords(p, str);
  8140. toXML(p, str, 0, XML_Format|XML_SortTags);
  8141. }
  8142. else
  8143. str.append("Unrecognized workunit format");
  8144. return str;
  8145. }
  8146. extern WORKUNIT_API void exportWorkUnitToXMLFile(const IConstWorkUnit *wu, const char * filename, unsigned extraXmlFlags, bool unpack, bool includeProgress, bool hidePasswords)
  8147. {
  8148. const IExtendedWUInterface *ewu = queryExtendedWU(wu);
  8149. if (ewu)
  8150. {
  8151. Linked<IPropertyTree> p;
  8152. if (unpack||includeProgress)
  8153. p.setown(ewu->getUnpackedTree(includeProgress));
  8154. else
  8155. p.set(ewu->queryPTree());
  8156. if (hidePasswords && p->hasProp("Variables/Variable[Format/@password]"))
  8157. return exportWorkUnitToXMLFileWithHiddenPasswords(p, filename, extraXmlFlags);
  8158. saveXML(filename, p, 0, XML_Format|XML_SortTags|extraXmlFlags);
  8159. }
  8160. else
  8161. throw makeStringException(0, "Unrecognized workunit format");
  8162. }
  8163. extern WORKUNIT_API void submitWorkUnit(const char *wuid, const char *username, const char *password)
  8164. {
  8165. MemoryBuffer buffer;
  8166. Owned<INamedQueueConnection> conn = createNamedQueueConnection(0); // MORE - security token?
  8167. Owned<IWorkUnitFactory> factory = getWorkUnitFactory();
  8168. Owned<IWorkUnit> workunit = factory->updateWorkUnit(wuid);
  8169. assertex(workunit);
  8170. SCMStringBuffer token;
  8171. createToken(wuid, username, password, token);
  8172. workunit->setSecurityToken(token.str());
  8173. StringAttr clusterName(workunit->queryClusterName());
  8174. if (!clusterName.length())
  8175. throw MakeStringException(WUERR_InvalidCluster, "No target cluster specified");
  8176. workunit->commit();
  8177. workunit.clear();
  8178. Owned<IConstWUClusterInfo> clusterInfo = getTargetClusterInfo(clusterName.str());
  8179. if (!clusterInfo)
  8180. throw MakeStringException(WUERR_InvalidCluster, "Unknown cluster %s", clusterName.str());
  8181. SCMStringBuffer serverQueue;
  8182. clusterInfo->getServerQueue(serverQueue);
  8183. assertex(serverQueue.length());
  8184. Owned<IJobQueue> queue = createJobQueue(serverQueue.str());
  8185. if (!queue.get())
  8186. throw MakeStringException(WUERR_InvalidQueue, "Could not create workunit queue");
  8187. IJobQueueItem *item = createJobQueueItem(wuid);
  8188. queue->enqueue(item);
  8189. }
  8190. extern WORKUNIT_API void abortWorkUnit(const char *wuid)
  8191. {
  8192. StringBuffer xpath("/WorkUnitAborts/");
  8193. xpath.append(wuid);
  8194. Owned<IRemoteConnection> acon = querySDS().connect(xpath.str(), myProcessSession(), RTM_LOCK_WRITE|RTM_CREATE, SDS_LOCK_TIMEOUT);
  8195. acon->queryRoot()->setPropInt(NULL, 1);
  8196. }
  8197. extern WORKUNIT_API void secSubmitWorkUnit(const char *wuid, ISecManager &secmgr, ISecUser &secuser)
  8198. {
  8199. if (checkWuSecAccess(wuid, &secmgr, &secuser, SecAccess_Write, "Submit", true, true))
  8200. submitWorkUnit(wuid, secuser.getName(), secuser.credentials().getPassword());
  8201. }
  8202. extern WORKUNIT_API void secAbortWorkUnit(const char *wuid, ISecManager &secmgr, ISecUser &secuser)
  8203. {
  8204. if (checkWuSecAccess(wuid, &secmgr, &secuser, SecAccess_Write, "Submit", true, true))
  8205. abortWorkUnit(wuid);
  8206. }
  8207. extern WORKUNIT_API void submitWorkUnit(const char *wuid, ISecManager *secmgr, ISecUser *secuser)
  8208. {
  8209. if (secmgr && secuser)
  8210. return secSubmitWorkUnit(wuid, *secmgr, *secuser);
  8211. if (secuser)
  8212. return submitWorkUnit(wuid, secuser->getName(), secuser->credentials().getPassword());
  8213. submitWorkUnit(wuid, "", "");
  8214. }
  8215. extern WORKUNIT_API void abortWorkUnit(const char *wuid, ISecManager *secmgr, ISecUser *secuser)
  8216. {
  8217. if (secmgr && secuser)
  8218. return secAbortWorkUnit(wuid, *secmgr, *secuser);
  8219. abortWorkUnit(wuid);
  8220. }
  8221. bool CLocalWorkUnit::hasWorkflow() const
  8222. {
  8223. return p->hasProp("Workflow");
  8224. }
  8225. unsigned CLocalWorkUnit::queryEventScheduledCount() const
  8226. {
  8227. CriticalBlock block(crit);
  8228. return p->getPropInt("Workflow/@eventScheduledCount", 0);
  8229. }
  8230. void CLocalWorkUnit::incEventScheduledCount()
  8231. {
  8232. CriticalBlock block(crit);
  8233. p->setPropInt("Workflow/@eventScheduledCount", p->getPropInt("Workflow/@eventScheduledCount", 0)+1);
  8234. }
  8235. IPropertyTree * CLocalWorkUnit::queryWorkflowTree() const
  8236. {
  8237. CriticalBlock block(crit);
  8238. return p->queryPropTree("Workflow");
  8239. }
  8240. IConstWorkflowItemIterator* CLocalWorkUnit::getWorkflowItems() const
  8241. {
  8242. // For this to be legally called, we must have the read-able interface. So we are already locked for (at least) read.
  8243. CriticalBlock block(crit);
  8244. if(!workflowIteratorCached)
  8245. {
  8246. assertex(!workflowIterator);
  8247. Owned<IPropertyTree> s = p->getPropTree("Workflow");
  8248. if(s)
  8249. workflowIterator.setown(createWorkflowItemIterator(s));
  8250. workflowIteratorCached = true;
  8251. }
  8252. return workflowIterator.getLink();
  8253. }
  8254. IWorkflowItemArray * CLocalWorkUnit::getWorkflowClone() const
  8255. {
  8256. unsigned count = 0;
  8257. Owned<IConstWorkflowItemIterator> iter = getWorkflowItems();
  8258. for(iter->first(); iter->isValid(); iter->next())
  8259. count++;
  8260. Owned<IWorkflowItemArray> array = createWorkflowItemArray(count);
  8261. for(iter->first(); iter->isValid(); iter->next())
  8262. array->addClone(iter->query());
  8263. return array.getLink();
  8264. }
  8265. IWorkflowItem * CLocalWorkUnit::addWorkflowItem(unsigned wfid, WFType type, WFMode mode, unsigned success, unsigned failure, unsigned recovery, unsigned retriesAllowed, unsigned contingencyFor)
  8266. {
  8267. // For this to be legally called, we must have the write-able interface. So we are already locked for write.
  8268. CriticalBlock block(crit);
  8269. workflowIterator.clear();
  8270. workflowIteratorCached = false;
  8271. IPropertyTree * s = p->queryPropTree("Workflow");
  8272. if(!s)
  8273. s = p->addPropTree("Workflow", createPTree("Workflow"));
  8274. return createWorkflowItem(s, wfid, type, mode, success, failure, recovery, retriesAllowed, contingencyFor);
  8275. }
  8276. IWorkflowItemIterator * CLocalWorkUnit::updateWorkflowItems()
  8277. {
  8278. // For this to be legally called, we must have the write-able interface. So we are already locked for write.
  8279. CriticalBlock block(crit);
  8280. if(!workflowIterator)
  8281. {
  8282. IPropertyTree * s = p->queryPropTree("Workflow");
  8283. if(!s)
  8284. s = p->addPropTree("Workflow", createPTree("Workflow"));
  8285. workflowIterator.setown(createWorkflowItemIterator(s));
  8286. workflowIteratorCached = true;
  8287. }
  8288. return workflowIterator.getLink();
  8289. }
  8290. void CLocalWorkUnit::syncRuntimeWorkflow(IWorkflowItemArray * array)
  8291. {
  8292. Owned<IWorkflowItemIterator> iter = updateWorkflowItems();
  8293. Owned<IWorkflowItem> item;
  8294. for(iter->first(); iter->isValid(); iter->next())
  8295. {
  8296. item.setown(iter->get());
  8297. item->syncRuntimeData(array->queryWfid(item->queryWfid()));
  8298. }
  8299. workflowIterator.clear();
  8300. workflowIteratorCached = false;
  8301. }
  8302. void CLocalWorkUnit::resetWorkflow()
  8303. {
  8304. if (hasWorkflow())
  8305. {
  8306. Owned<IWorkflowItemIterator> iter = updateWorkflowItems();
  8307. Owned<IWorkflowItem> wf;
  8308. for(iter->first(); iter->isValid(); iter->next())
  8309. {
  8310. wf.setown(iter->get());
  8311. wf->reset();
  8312. }
  8313. workflowIterator.clear();
  8314. workflowIteratorCached = false;
  8315. }
  8316. }
  8317. void CLocalWorkUnit::schedule()
  8318. {
  8319. CriticalBlock block(crit);
  8320. if(queryEventScheduledCount() == 0) return;
  8321. switch(getState())
  8322. {
  8323. case WUStateCompleted:
  8324. setState(WUStateWait);
  8325. break;
  8326. case WUStateFailed:
  8327. case WUStateArchived:
  8328. case WUStateAborting:
  8329. case WUStateAborted:
  8330. case WUStateScheduled:
  8331. throw MakeStringException(WUERR_CannotSchedule, "Cannot schedule workunit in this state");
  8332. }
  8333. StringBuffer rootPath;
  8334. rootPath.append("/Schedule/").append(queryClusterName());
  8335. Owned<IRemoteConnection> conn = querySDS().connect(rootPath.str(), myProcessSession(), RTM_LOCK_WRITE | RTM_CREATE_QUERY, SDS_LOCK_TIMEOUT);
  8336. Owned<IPropertyTree> root = conn->getRoot();
  8337. if(!root->hasChildren())
  8338. {
  8339. StringBuffer addPath;
  8340. addPath.append("/Schedulers/").append(queryClusterName());
  8341. Owned<IRemoteConnection> addConn = querySDS().connect(addPath.str(), myProcessSession(), RTM_LOCK_WRITE | RTM_CREATE_QUERY, SDS_LOCK_TIMEOUT);
  8342. }
  8343. char const * wuid = p->queryName();
  8344. StringBuffer xpath("*/*/");
  8345. ncnameEscape(wuid, xpath);
  8346. bool more;
  8347. do more = root->removeProp(xpath.str()); while(more);
  8348. Owned<IConstWorkflowItemIterator> iter = getWorkflowItems();
  8349. Owned<IWorkflowEvent> event;
  8350. Owned<IPropertyTree> branch1, branch2;
  8351. for(iter->first(); iter->isValid(); iter->next())
  8352. {
  8353. event.setown(iter->query()->getScheduleEvent());
  8354. if(!event) continue;
  8355. ncnameEscape(event->queryName(), xpath.clear());
  8356. ensurePTree(root, xpath.str());
  8357. branch1.setown(root->getPropTree(xpath.str()));
  8358. ncnameEscape(event->queryText(), xpath.clear());
  8359. ensurePTree(branch1, xpath.str());
  8360. branch2.setown(branch1->getPropTree(xpath.str()));
  8361. ncnameEscape(wuid, xpath.clear());
  8362. ensurePTree(branch2, xpath.str());
  8363. }
  8364. }
  8365. void CLocalWorkUnit::deschedule()
  8366. {
  8367. if(queryEventScheduledCount() == 0) return;
  8368. if(getState() == WUStateWait)
  8369. setState(WUStateCompleted);
  8370. doDescheduleWorkkunit(p->queryName());
  8371. }
  8372. mapEnums localFileUploadTypes[] = {
  8373. { UploadTypeFileSpray, "FileSpray" },
  8374. { UploadTypeWUResult, "WUResult" },
  8375. { UploadTypeWUResultCsv, "WUResultCsv" },
  8376. { UploadTypeWUResultXml, "WUResultXml" },
  8377. { UploadTypeSize, NULL }
  8378. };
  8379. class CLocalFileUpload : public CInterface, implements IConstLocalFileUpload
  8380. {
  8381. public:
  8382. CLocalFileUpload(IPropertyTree * _tree) : tree(_tree) {}
  8383. CLocalFileUpload(unsigned id, LocalFileUploadType type, char const * source, char const * destination, char const * eventTag)
  8384. {
  8385. tree.setown(createPTree());
  8386. tree->setPropInt("@id", id);
  8387. setEnum(tree, "@type", type, localFileUploadTypes);
  8388. tree->setProp("@source", source);
  8389. tree->setProp("@destination", destination);
  8390. if (eventTag)
  8391. tree->setProp("@eventTag", eventTag);
  8392. }
  8393. IMPLEMENT_IINTERFACE;
  8394. IPropertyTree * getTree() { return tree.getLink(); }
  8395. virtual unsigned queryID() const { return tree->getPropInt("@id"); }
  8396. virtual LocalFileUploadType queryType() const { return (LocalFileUploadType)getEnum(tree, "@type", localFileUploadTypes); }
  8397. virtual IStringVal & getSource(IStringVal & ret) const { ret.set(tree->queryProp("@source")); return ret; }
  8398. virtual IStringVal & getDestination(IStringVal & ret) const { ret.set(tree->queryProp("@destination")); return ret; }
  8399. virtual IStringVal & getEventTag(IStringVal & ret) const { if(tree->hasProp("@eventTag")) ret.set(tree->queryProp("@eventTag")); else ret.clear(); return ret; }
  8400. private:
  8401. Owned<IPropertyTree> tree;
  8402. };
  8403. class CLocalFileUploadIterator : public CInterface, implements IConstLocalFileUploadIterator
  8404. {
  8405. public:
  8406. CLocalFileUploadIterator(IPropertyTree * _tree) : tree(_tree), iter(tree->getElements("LocalFileUpload")) {}
  8407. IMPLEMENT_IINTERFACE;
  8408. bool first() { return iter->first(); }
  8409. bool isValid() { return iter->isValid(); }
  8410. bool next() { return iter->next(); }
  8411. IConstLocalFileUpload * get() { return new CLocalFileUpload(&iter->get()); }
  8412. private:
  8413. Owned<IPropertyTree> tree;
  8414. Owned<IPropertyTreeIterator> iter;
  8415. };
  8416. IConstLocalFileUploadIterator * CLocalWorkUnit::getLocalFileUploads() const
  8417. {
  8418. // For this to be legally called, we must have the read-able interface. So we are already locked for (at least) read.
  8419. CriticalBlock block(crit);
  8420. Owned<IPropertyTree> s = p->getPropTree("LocalFileUploads");
  8421. if(s)
  8422. return new CLocalFileUploadIterator(s.getClear());
  8423. else
  8424. return NULL;
  8425. }
  8426. bool CLocalWorkUnit::requiresLocalFileUpload() const
  8427. {
  8428. SCMStringBuffer dest;
  8429. Owned<IConstWUResult> result;
  8430. Owned<IConstLocalFileUploadIterator> iter(getLocalFileUploads());
  8431. if(!iter)
  8432. return false;
  8433. for(iter->first(); iter->isValid(); iter->next())
  8434. {
  8435. Owned<IConstLocalFileUpload> upload(iter->get());
  8436. switch(upload->queryType())
  8437. {
  8438. case UploadTypeWUResult:
  8439. case UploadTypeWUResultCsv:
  8440. case UploadTypeWUResultXml:
  8441. upload->getDestination(dest);
  8442. result.setown(getResultByName(dest.str()));
  8443. if(!result)
  8444. return true;
  8445. break;
  8446. default:
  8447. throw MakeStringException(WUERR_InvalidUploadFormat, "Unsupported local file upload type %s", getEnumText(upload->queryType(), localFileUploadTypes));
  8448. }
  8449. }
  8450. return false;
  8451. }
  8452. unsigned CLocalWorkUnit::addLocalFileUpload(LocalFileUploadType type, char const * source, char const * destination, char const * eventTag)
  8453. {
  8454. // For this to be legally called, we must have the write-able interface. So we are already locked for write.
  8455. CriticalBlock block(crit);
  8456. IPropertyTree * s = p->queryPropTree("LocalFileUploads");
  8457. if(!s)
  8458. s = p->addPropTree("LocalFileUploads", createPTree());
  8459. unsigned id = s->numChildren();
  8460. Owned<CLocalFileUpload> upload = new CLocalFileUpload(id, type, source, destination, eventTag);
  8461. s->addPropTree("LocalFileUpload", upload->getTree());
  8462. return id;
  8463. }
  8464. #if 0
  8465. void testConstWorkflow(IConstWorkflowItem * cwf, bool * okay, bool * dep)
  8466. {
  8467. DBGLOG("Test workflow const iface %u", cwf->queryWfid());
  8468. unsigned deps = 0;
  8469. Owned<IWorkflowDependencyIterator> diter;
  8470. switch(cwf->queryWfid())
  8471. {
  8472. case 1:
  8473. assertex(!cwf->isScheduled());
  8474. assertex(cwf->queryType() == WFTypeNormal);
  8475. assertex(cwf->queryState() == WFStateNull);
  8476. diter.setown(cwf->getDependencies());
  8477. for(diter->first(); diter->isValid(); diter->next())
  8478. deps++;
  8479. assertex(deps==0);
  8480. okay[0] = true;
  8481. break;
  8482. case 2:
  8483. assertex(!cwf->isScheduled());
  8484. assertex(cwf->queryType() == WFTypeRecovery);
  8485. assertex(cwf->queryState() == WFStateSkip);
  8486. okay[1] = true;
  8487. break;
  8488. case 3:
  8489. assertex(cwf->queryContingencyFor() == 4);
  8490. okay[2] = true;
  8491. break;
  8492. case 4:
  8493. assertex(cwf->isScheduled());
  8494. assertex(cwf->queryType() == WFTypeNormal);
  8495. assertex(cwf->queryState() == WFStateReqd);
  8496. assertex(cwf->querySuccess() == 0);
  8497. assertex(cwf->queryFailure() == 3);
  8498. assertex(cwf->queryRecovery() == 2);
  8499. assertex(cwf->queryRetriesAllowed() == 10);
  8500. assertex(cwf->queryRetriesRemaining() == 10);
  8501. diter.setown(cwf->getDependencies());
  8502. for(diter->first(); diter->isValid(); diter->next())
  8503. {
  8504. dep[diter->query()-1] = true;
  8505. deps++;
  8506. }
  8507. assertex(deps==2);
  8508. assertex(dep[0]);
  8509. assertex(dep[1]);
  8510. okay[3] = true;
  8511. break;
  8512. case 5:
  8513. assertex(cwf->isScheduled());
  8514. assertex(!cwf->isScheduledNow());
  8515. assertex(cwf->querySchedulePriority() == 75);
  8516. assertex(cwf->queryScheduleCount() == 5);
  8517. assertex(cwf->queryScheduleCountRemaining() == 5);
  8518. okay[4] = true;
  8519. break;
  8520. case 6:
  8521. assertex(cwf->isScheduled());
  8522. assertex(!cwf->isScheduledNow());
  8523. assertex(cwf->querySchedulePriority() == 25);
  8524. assertex(!cwf->hasScheduleCount());
  8525. okay[5] = true;
  8526. break;
  8527. default:
  8528. assertex(!"unknown wfid in test");
  8529. }
  8530. }
  8531. void testRuntimeWorkflow(IRuntimeWorkflowItem * rwf, bool * okay)
  8532. {
  8533. DBGLOG("Test workflow runtime iface %u", rwf->queryWfid());
  8534. switch(rwf->queryWfid())
  8535. {
  8536. case 1:
  8537. case 2:
  8538. case 3:
  8539. okay[rwf->queryWfid()-1] = true;
  8540. break;
  8541. case 4:
  8542. {
  8543. unsigned tries = 0;
  8544. while(rwf->testAndDecRetries())
  8545. tries++;
  8546. assertex(tries == 10);
  8547. assertex(rwf->queryRetriesRemaining() == 0);
  8548. rwf->setState(WFStateFail);
  8549. assertex(rwf->queryState() == WFStateFail);
  8550. rwf->reset();
  8551. assertex(rwf->queryRetriesRemaining() == 10);
  8552. assertex(rwf->queryState() == WFStateReqd);
  8553. }
  8554. okay[3] = true;
  8555. break;
  8556. case 5:
  8557. {
  8558. assertex(rwf->queryScheduleCountRemaining() == 5);
  8559. unsigned count = 0;
  8560. do count++; while(rwf->decAndTestScheduleCountRemaining());
  8561. assertex(count == 5);
  8562. assertex(rwf->queryScheduleCountRemaining() == 0);
  8563. rwf->reset();
  8564. assertex(rwf->queryScheduleCountRemaining() == 5);
  8565. }
  8566. okay[4] = true;
  8567. break;
  8568. case 6:
  8569. {
  8570. assertex(!rwf->hasScheduleCount());
  8571. unsigned count;
  8572. for(count=0; count<20; count++)
  8573. assertex(rwf->decAndTestScheduleCountRemaining());
  8574. }
  8575. okay[5] = true;
  8576. break;
  8577. default:
  8578. assertex(!"unknown wfid in test");
  8579. }
  8580. }
  8581. void testWorkflow()
  8582. {
  8583. DBGLOG("workunit.cpp : testWorkflow");
  8584. CLocalWorkUnit wu("W-WF-TEST", 0, 0, 0);
  8585. Owned<IWorkflowItem> wf;
  8586. wf.setown(wu.addWorkflowItem(1, WFTypeNormal, 0, 0, 0, 0, 0));
  8587. wf.setown(wu.addWorkflowItem(2, WFTypeRecovery, 0, 0, 0, 0, 0));
  8588. wf.setown(wu.addWorkflowItem(3, WFTypeFailure, 0, 0, 0, 0, 4));
  8589. wf.setown(wu.addWorkflowItem(4, WFTypeNormal, 0, 3, 2, 10, 0));
  8590. wf->setScheduledNow();
  8591. wf->addDependency(1);
  8592. wf.setown(wu.addWorkflowItem(5, WFTypeNormal, 0, 0, 0, 0, 0));
  8593. wf->setScheduledOn("test", "foo*");
  8594. wf->setSchedulePriority(75);
  8595. wf->setScheduleCount(5);
  8596. wf.setown(wu.addWorkflowItem(6, WFTypeNormal, 0, 0, 0, 0, 0));
  8597. wf->setScheduledOn("test", "bar*");
  8598. wf->setSchedulePriority(25);
  8599. unsigned const n = 6;
  8600. bool okay[n];
  8601. bool dep[n];
  8602. unsigned i;
  8603. for(i=0; i<n; i++)
  8604. okay[i] = dep[i] = 0;
  8605. Owned<IConstWorkflowItemIterator> citer(wu.getWorkflowItems());
  8606. for(citer->first(); citer->isValid(); citer->next())
  8607. testConstWorkflow(citer->query(), okay, dep);
  8608. for(i=0; i<n; i++)
  8609. {
  8610. assertex(okay[i]);
  8611. okay[i] = false;
  8612. }
  8613. Owned<IWorkflowItemIterator> miter(wu.updateWorkflowItems());
  8614. for(miter->first(); miter->isValid(); miter->next())
  8615. {
  8616. Owned<IRuntimeWorkflowItem> rwf(miter->get());
  8617. testRuntimeWorkflow(rwf, okay);
  8618. }
  8619. for(i=0; i<n; i++)
  8620. {
  8621. assertex(okay[i]);
  8622. okay[i] = dep[i] = false;
  8623. }
  8624. Owned<IWorkflowItemArray> array(wu.getWorkflowClone());
  8625. unsigned wfid;
  8626. for(wfid = 1; array->isValid(wfid); wfid++)
  8627. testConstWorkflow(&array->queryWfid(wfid), okay, dep);
  8628. for(i=0; i<n; i++)
  8629. {
  8630. assertex(okay[i]);
  8631. okay[i] = false;
  8632. }
  8633. for(wfid = 1; array->isValid(wfid); wfid++)
  8634. testRuntimeWorkflow(&array->queryWfid(wfid), okay);
  8635. for(i=0; i<n; i++)
  8636. {
  8637. assertex(okay[i]);
  8638. okay[i] = false;
  8639. }
  8640. }
  8641. #endif
  8642. //------------------------------------------------------------------------------------------
  8643. extern WUState waitForWorkUnitToComplete(const char * wuid, int timeout, bool returnOnWaitState)
  8644. {
  8645. return factory->waitForWorkUnit(wuid, (unsigned) timeout, false, returnOnWaitState);
  8646. }
  8647. extern WORKUNIT_API WUState secWaitForWorkUnitToComplete(const char * wuid, ISecManager &secmgr, ISecUser &secuser, int timeout, bool returnOnWaitState)
  8648. {
  8649. if (checkWuSecAccess(wuid, &secmgr, &secuser, SecAccess_Read, "Wait for Complete", false, true))
  8650. return waitForWorkUnitToComplete(wuid, timeout, returnOnWaitState);
  8651. return WUStateUnknown;
  8652. }
  8653. extern bool waitForWorkUnitToCompile(const char * wuid, int timeout)
  8654. {
  8655. switch(factory->waitForWorkUnit(wuid, (unsigned) timeout, true, true))
  8656. {
  8657. case WUStateCompiled:
  8658. case WUStateCompleted:
  8659. case WUStateWait:
  8660. case WUStateUploadingFiles:
  8661. return true;
  8662. default:
  8663. return false;
  8664. }
  8665. }
  8666. extern WORKUNIT_API bool secWaitForWorkUnitToCompile(const char * wuid, ISecManager &secmgr, ISecUser &secuser, int timeout)
  8667. {
  8668. if (checkWuSecAccess(wuid, &secmgr, &secuser, SecAccess_Read, "Wait for Compile", false, true))
  8669. return waitForWorkUnitToCompile(wuid, timeout);
  8670. return false;
  8671. }
  8672. extern WORKUNIT_API bool secDebugWorkunit(const char * wuid, ISecManager &secmgr, ISecUser &secuser, const char *command, StringBuffer &response)
  8673. {
  8674. if (strnicmp(command, "<debug:", 7) == 0 && checkWuSecAccess(wuid, &secmgr, &secuser, SecAccess_Read, "Debug", false, true))
  8675. {
  8676. Owned<IConstWorkUnit> wu = factory->openWorkUnit(wuid, false, &secmgr, &secuser);
  8677. SCMStringBuffer ip;
  8678. unsigned port;
  8679. port = wu->getDebugAgentListenerPort();
  8680. wu->getDebugAgentListenerIP(ip);
  8681. SocketEndpoint debugEP(ip.str(), port);
  8682. Owned<ISocket> socket = ISocket::connect_timeout(debugEP, 1000);
  8683. unsigned len = (size32_t)strlen(command);
  8684. unsigned revlen = len;
  8685. _WINREV(revlen);
  8686. socket->write(&revlen, sizeof(revlen));
  8687. socket->write(command, len);
  8688. for (;;)
  8689. {
  8690. socket->read(&len, sizeof(len));
  8691. _WINREV(len);
  8692. if (len == 0)
  8693. break;
  8694. if (len & 0x80000000)
  8695. {
  8696. throwUnexpected();
  8697. }
  8698. char * mem = (char*) response.reserve(len);
  8699. socket->read(mem, len);
  8700. }
  8701. return true;
  8702. }
  8703. return false;
  8704. }
  8705. void updateSuppliedXmlParams(IWorkUnit * w)
  8706. {
  8707. Owned<const IPropertyTree> params = w->getXmlParams();
  8708. if (!params)
  8709. return;
  8710. Owned<IPropertyTreeIterator> elems = params->getElements("*");
  8711. ForEach(*elems)
  8712. {
  8713. IPropertyTree & curVal = elems->query();
  8714. const char *name = curVal.queryName();
  8715. Owned<IWUResult> r = updateWorkUnitResult(w, name, -1);
  8716. if (r)
  8717. {
  8718. StringBuffer s;
  8719. if (r->isResultScalar() && !curVal.hasChildren())
  8720. {
  8721. curVal.getProp(".", s);
  8722. r->setResultXML(s);
  8723. r->setResultStatus(ResultStatusSupplied);
  8724. }
  8725. else
  8726. {
  8727. toXML(&curVal, s);
  8728. bool isSet = (curVal.hasProp("Item") || curVal.hasProp("string"));
  8729. r->setResultRaw(s.length(), s.str(), isSet ? ResultFormatXmlSet : ResultFormatXml);
  8730. }
  8731. }
  8732. else
  8733. DBGLOG("WARNING: no matching variable in workunit for input parameter %s", name);
  8734. }
  8735. }
  8736. IWUResult * updateWorkUnitResult(IWorkUnit * w, const char *name, unsigned sequence)
  8737. {
  8738. switch ((int)sequence)
  8739. {
  8740. case ResultSequenceStored:
  8741. return w->updateVariableByName(name);
  8742. case ResultSequencePersist:
  8743. return w->updateGlobalByName(name);
  8744. case ResultSequenceInternal:
  8745. case ResultSequenceOnce:
  8746. return w->updateTemporaryByName(name);
  8747. default:
  8748. return w->updateResultBySequence(sequence);
  8749. }
  8750. }
  8751. IConstWUResult * getWorkUnitResult(IConstWorkUnit * w, const char *name, unsigned sequence)
  8752. {
  8753. switch ((int)sequence)
  8754. {
  8755. case ResultSequenceStored:
  8756. return w->getVariableByName(name);
  8757. case ResultSequencePersist:
  8758. return w->getGlobalByName(name);
  8759. case ResultSequenceInternal:
  8760. case ResultSequenceOnce:
  8761. return w->getTemporaryByName(name);
  8762. default:
  8763. if (name && name[0])
  8764. return w->getResultByName(name);//name takes precedence over sequence
  8765. else
  8766. return w->getResultBySequence(sequence);
  8767. }
  8768. }
  8769. extern WORKUNIT_API bool getWorkUnitCreateTime(const char *wuid,CDateTime &time)
  8770. {
  8771. if (wuid) {
  8772. char prefchar;
  8773. unsigned year,month,day,hour,min,sec;
  8774. if (sscanf(wuid, "%c%4u%2u%2u-%2u%2u%2u", &prefchar, &year, &month, &day, &hour, &min, &sec)==7) {
  8775. time.set(year, month, day, hour, min, sec, 0, true);
  8776. // time.setDate(year, month, day);
  8777. // time.setTime(hour, min, sec, 0, true); // for some reason time is local
  8778. return true;
  8779. }
  8780. }
  8781. return false;
  8782. }
  8783. extern WORKUNIT_API IStringVal& createToken(const char *wuid, const char *user, const char *password, IStringVal &str)
  8784. {
  8785. StringBuffer wu, token("X");
  8786. wu.append(wuid).append(';').append(user).append(';').append(password);
  8787. encrypt(token,wu.str());
  8788. str.set(token.str());
  8789. return str;
  8790. }
  8791. // This will be replaced by something more secure!
  8792. extern WORKUNIT_API void extractToken(const char *token, const char *wuid, IStringVal &user, IStringVal &password)
  8793. {
  8794. if (token && *token)
  8795. {
  8796. StringBuffer wu;
  8797. decrypt(wu, token+1);
  8798. const char *finger = strchr(wu.str(),';');
  8799. if (finger && strnicmp(wuid, wu.str(), finger-wu.str())==0)
  8800. {
  8801. const char *finger1 = strchr(++finger,';');
  8802. if(finger1)
  8803. {
  8804. user.setLen(finger, (size32_t)(finger1-finger));
  8805. finger1++;
  8806. password.setLen(finger1, (size32_t)(wu.str() + wu.length() - finger1));
  8807. return;
  8808. }
  8809. }
  8810. throw MakeStringException(WUERR_InvalidSecurityToken, "Invalid call to extractToken");
  8811. }
  8812. }
  8813. extern WORKUNIT_API WUState getWorkUnitState(const char* state)
  8814. {
  8815. return (WUState) getEnum(state, states);
  8816. }
  8817. const LogMsgCategory MCschedconn = MCprogress(1000); // Category used to inform about schedule synchronization
  8818. class CWorkflowScheduleConnection : public CInterface, implements IWorkflowScheduleConnection
  8819. {
  8820. public:
  8821. CWorkflowScheduleConnection(char const * wuid)
  8822. {
  8823. basexpath.append("/WorkflowSchedule/").append(wuid);
  8824. flagxpath.append(basexpath.str()).append("/Active");
  8825. }
  8826. IMPLEMENT_IINTERFACE;
  8827. virtual void lock()
  8828. {
  8829. LOG(MCschedconn, "Locking base schedule connection");
  8830. baseconn.setown(querySDS().connect(basexpath.str(), myProcessSession(), RTM_CREATE_QUERY | RTM_LOCK_WRITE, INFINITE));
  8831. if(!baseconn)
  8832. throw MakeStringException(WUERR_ScheduleLockFailed, "Could not get base workflow schedule lock");
  8833. }
  8834. virtual void unlock()
  8835. {
  8836. LOG(MCschedconn, "Unlocking base schedule connection");
  8837. baseconn.clear();
  8838. }
  8839. virtual void setActive()
  8840. {
  8841. LOG(MCschedconn, "Setting active flag in schedule connection");
  8842. flagconn.setown(querySDS().connect(flagxpath.str(), myProcessSession(), RTM_CREATE | RTM_LOCK_WRITE | RTM_DELETE_ON_DISCONNECT, INFINITE));
  8843. if(!flagconn)
  8844. throw MakeStringException(WUERR_ScheduleLockFailed, "Could not get active workflow schedule lock");
  8845. }
  8846. virtual void resetActive()
  8847. {
  8848. LOG(MCschedconn, "Resetting active flag in schedule connection");
  8849. flagconn.clear();
  8850. }
  8851. virtual bool queryActive()
  8852. {
  8853. return baseconn->queryRoot()->hasProp("Active");
  8854. }
  8855. virtual bool pull(IWorkflowItemArray * workflow)
  8856. {
  8857. assertex(baseconn);
  8858. Owned<IPropertyTree> root = baseconn->getRoot();
  8859. Owned<IPropertyTree> eventQueue = root->getPropTree("EventQueue");
  8860. if(!eventQueue) return false;
  8861. if(!eventQueue->hasProp("Item")) return false;
  8862. {
  8863. Owned<IPropertyTreeIterator> eventItems = eventQueue->getElements("Item");
  8864. Owned<IPropertyTree> eventItem;
  8865. Owned<IRuntimeWorkflowItemIterator> wfItems = workflow->getSequenceIterator();
  8866. Owned<IRuntimeWorkflowItem> wfItem;
  8867. for(eventItems->first(); eventItems->isValid(); eventItems->next())
  8868. {
  8869. eventItem.setown(&eventItems->get());
  8870. const char * eventName = eventItem->queryProp("@name");
  8871. const char * eventText = eventItem->queryProp("@text");
  8872. for(wfItems->first(); wfItems->isValid(); wfItems->next())
  8873. {
  8874. wfItem.setown(wfItems->get());
  8875. if(wfItem->queryState() != WFStateWait)
  8876. continue;
  8877. Owned<IWorkflowEvent> targetEvent = wfItem->getScheduleEvent();
  8878. if(!targetEvent || !targetEvent->matches(eventName, eventText))
  8879. continue;
  8880. wfItem->setEvent(eventName, eventText);
  8881. wfItem->setState(WFStateReqd);
  8882. resetDependentsState(workflow, *wfItem);
  8883. }
  8884. }
  8885. }
  8886. bool more;
  8887. do
  8888. more = eventQueue->removeProp("Item");
  8889. while(more);
  8890. return true;
  8891. }
  8892. virtual void push(char const * name, char const * text)
  8893. {
  8894. assertex(baseconn);
  8895. Owned<IPropertyTree> root = baseconn->getRoot();
  8896. ensurePTree(root, "EventQueue");
  8897. Owned<IPropertyTree> eventQueue = root->getPropTree("EventQueue");
  8898. Owned<IPropertyTree> eventItem = createPTree();
  8899. eventItem->setProp("@name", name);
  8900. eventItem->setProp("@text", text);
  8901. eventQueue->addPropTree("Item", eventItem.getLink());
  8902. }
  8903. virtual void remove()
  8904. {
  8905. if (baseconn)
  8906. {
  8907. baseconn->close(true);
  8908. baseconn.clear();
  8909. }
  8910. }
  8911. private:
  8912. void resetItemStateAndDependents(IWorkflowItemArray * workflow, unsigned wfid) const
  8913. {
  8914. if (wfid)
  8915. resetItemStateAndDependents(workflow, workflow->queryWfid(wfid));
  8916. }
  8917. void resetItemStateAndDependents(IWorkflowItemArray * workflow, IRuntimeWorkflowItem & item) const
  8918. {
  8919. switch(item.queryState())
  8920. {
  8921. case WFStateDone:
  8922. case WFStateFail:
  8923. {
  8924. item.setState(WFStateNull);
  8925. resetItemStateAndDependents(workflow, item.queryPersistWfid());
  8926. resetDependentsState(workflow, item);
  8927. break;
  8928. }
  8929. }
  8930. }
  8931. void resetDependentsState(IWorkflowItemArray * workflow, IRuntimeWorkflowItem & item) const
  8932. {
  8933. Owned<IWorkflowDependencyIterator> iter(item.getDependencies());
  8934. for(iter->first(); iter->isValid(); iter->next())
  8935. {
  8936. IRuntimeWorkflowItem & dep = workflow->queryWfid(iter->query());
  8937. resetItemStateAndDependents(workflow, dep);
  8938. }
  8939. }
  8940. private:
  8941. StringBuffer basexpath;
  8942. StringBuffer flagxpath;
  8943. Owned<IRemoteConnection> baseconn;
  8944. Owned<IRemoteConnection> flagconn;
  8945. };
  8946. extern WORKUNIT_API IWorkflowScheduleConnection * getWorkflowScheduleConnection(char const * wuid)
  8947. {
  8948. return new CWorkflowScheduleConnection(wuid);
  8949. }
  8950. extern WORKUNIT_API IExtendedWUInterface * queryExtendedWU(IConstWorkUnit * wu)
  8951. {
  8952. return QUERYINTERFACE(wu, IExtendedWUInterface);
  8953. }
  8954. extern WORKUNIT_API const IExtendedWUInterface * queryExtendedWU(const IConstWorkUnit * wu)
  8955. {
  8956. return QUERYINTERFACE(wu, const IExtendedWUInterface);
  8957. }
  8958. extern WORKUNIT_API void addExceptionToWorkunit(IWorkUnit * wu, ErrorSeverity severity, const char * source, unsigned code, const char * text, const char * filename, unsigned lineno, unsigned column)
  8959. {
  8960. Owned<IWUException> we = wu->createException();
  8961. we->setSeverity(severity);
  8962. we->setExceptionMessage(text);
  8963. if (source)
  8964. we->setExceptionSource(source);
  8965. if (code)
  8966. we->setExceptionCode(code);
  8967. if (filename)
  8968. we->setExceptionFileName(filename);
  8969. if (lineno)
  8970. {
  8971. we->setExceptionLineNo(lineno);
  8972. if (column)
  8973. we->setExceptionColumn(lineno);
  8974. }
  8975. }
  8976. const char * skipLeadingXml(const char * text)
  8977. {
  8978. if (!text)
  8979. return NULL;
  8980. //skip utf8 BOM, probably excessive
  8981. if (memcmp(text, UTF8_BOM, 3) == 0)
  8982. text += 3;
  8983. loop
  8984. {
  8985. if (isspace(*text))
  8986. text++;
  8987. else if (text[0] == '<' && text[1] == '?')
  8988. {
  8989. text += 2;
  8990. loop
  8991. {
  8992. if (!*text) break;
  8993. if (text[0] == '?' && text[1] == '>')
  8994. {
  8995. text += 2;
  8996. break;
  8997. }
  8998. text++;
  8999. }
  9000. }
  9001. else if (text[0] == '<' && text[1] == '!' && text[2] == '-' && text[3] == '-')
  9002. {
  9003. text += 4;
  9004. loop
  9005. {
  9006. if (!*text) break;
  9007. if (text[0] == '-' && text[1] == '-' && text[2] == '>')
  9008. {
  9009. text += 3;
  9010. break;
  9011. }
  9012. text++;
  9013. }
  9014. }
  9015. else
  9016. break;
  9017. }
  9018. return text;
  9019. }
  9020. extern WORKUNIT_API bool isArchiveQuery(const char * text)
  9021. {
  9022. text = skipLeadingXml(text);
  9023. if (!text)
  9024. return false;
  9025. const char * archivePrefix = "<Archive";
  9026. return memicmp(text, archivePrefix, strlen(archivePrefix)) == 0;
  9027. }
  9028. extern WORKUNIT_API bool isQueryManifest(const char * text)
  9029. {
  9030. text = skipLeadingXml(text);
  9031. if (!text)
  9032. return false;
  9033. const char * manifestPrefix = "<Manifest";
  9034. return memicmp(text, manifestPrefix, strlen(manifestPrefix)) == 0;
  9035. }
  9036. //------------------------------------------------------------------------------
  9037. // Named Alias helper function
  9038. static IPropertyTree * resolveQueryByDll(IPropertyTree * queryRegistry, const char * dll)
  9039. {
  9040. StringBuffer xpath;
  9041. xpath.append("Query[@dll=\"").append(dll).append("\"]");
  9042. return queryRegistry->getPropTree(xpath);
  9043. }
  9044. static IPropertyTree * resolveQueryByWuid(IPropertyTree * queryRegistry, const char * wuid)
  9045. {
  9046. StringBuffer xpath;
  9047. xpath.append("Query[@wuid=\"").append(wuid).append("\"]");
  9048. return queryRegistry->getPropTree(xpath);
  9049. }
  9050. static void clearAliases(IPropertyTree * queryRegistry, const char * id)
  9051. {
  9052. StringBuffer lcId(id);
  9053. lcId.toLowerCase();
  9054. StringBuffer xpath;
  9055. xpath.append("Alias[@id=\"").append(lcId).append("\"]");
  9056. Owned<IPropertyTreeIterator> iter = queryRegistry->getElements(xpath);
  9057. ForEach(*iter)
  9058. {
  9059. queryRegistry->removeProp(xpath.str());
  9060. }
  9061. }
  9062. IPropertyTree * addNamedQuery(IPropertyTree * queryRegistry, const char * name, const char * wuid, const char * dll, bool library, const char *userid, const char *snapshot)
  9063. {
  9064. StringBuffer lcName(name);
  9065. lcName.toLowerCase();
  9066. StringBuffer xpath;
  9067. xpath.append("Query[@name=\"").append(lcName.str()).append("\"]");
  9068. Owned<IPropertyTreeIterator> iter = queryRegistry->getElements(xpath);
  9069. unsigned seq = 1;
  9070. ForEach(*iter)
  9071. {
  9072. IPropertyTree &item = iter->query();
  9073. const char *thisWuid = item.queryProp("@wuid");
  9074. if (strieq(wuid, thisWuid))
  9075. return &item;
  9076. unsigned thisSeq = item.getPropInt("@seq");
  9077. if (thisSeq >= seq)
  9078. seq = thisSeq + 1;
  9079. }
  9080. StringBuffer id;
  9081. id.append(lcName).append(".").append(seq);
  9082. IPropertyTree * newEntry = createPTree("Query", ipt_caseInsensitive);
  9083. newEntry->setProp("@name", lcName);
  9084. newEntry->setProp("@wuid", wuid);
  9085. newEntry->setProp("@dll", dll);
  9086. newEntry->setProp("@id", id);
  9087. newEntry->setPropInt("@seq", seq);
  9088. if (library)
  9089. newEntry->setPropBool("@isLibrary", true);
  9090. if (userid && *userid)
  9091. newEntry->setProp("@publishedBy", userid);
  9092. if (snapshot && *snapshot)
  9093. newEntry->setProp("@snapshot", snapshot);
  9094. return queryRegistry->addPropTree("Query", newEntry);
  9095. }
  9096. void removeNamedQuery(IPropertyTree * queryRegistry, const char * id)
  9097. {
  9098. StringBuffer lcId(id);
  9099. lcId.toLowerCase();
  9100. clearAliases(queryRegistry, lcId);
  9101. StringBuffer xpath;
  9102. xpath.append("Query[@id=\"").append(lcId).append("\"]");
  9103. queryRegistry->removeProp(xpath);
  9104. }
  9105. void removeDllFromNamedQueries(IPropertyTree * queryRegistry, const char * dll)
  9106. {
  9107. Owned<IPropertyTree> match = resolveQueryByDll(queryRegistry, dll);
  9108. if (!match)
  9109. return;
  9110. clearAliases(queryRegistry, match->queryProp("@id"));
  9111. queryRegistry->removeTree(match);
  9112. }
  9113. void removeWuidFromNamedQueries(IPropertyTree * queryRegistry, const char * wuid)
  9114. {
  9115. Owned<IPropertyTree> match = resolveQueryByWuid(queryRegistry, wuid);
  9116. if (!match)
  9117. return;
  9118. clearAliases(queryRegistry, match->queryProp("@id"));
  9119. queryRegistry->removeTree(match);
  9120. }
  9121. void removeAliasesFromNamedQuery(IPropertyTree * queryRegistry, const char * id)
  9122. {
  9123. clearAliases(queryRegistry, id);
  9124. }
  9125. void setQueryAlias(IPropertyTree * queryRegistry, const char * name, const char * value)
  9126. {
  9127. StringBuffer lcName(name);
  9128. lcName.toLowerCase();
  9129. StringBuffer xpath;
  9130. xpath.append("Alias[@name=\"").append(lcName).append("\"]");
  9131. IPropertyTree * match = queryRegistry->queryPropTree(xpath);
  9132. if (!match)
  9133. {
  9134. IPropertyTree * newEntry = createPTree("Alias");
  9135. newEntry->setProp("@name", lcName);
  9136. match = queryRegistry->addPropTree("Alias", newEntry);
  9137. }
  9138. match->setProp("@id", value);
  9139. }
  9140. extern WORKUNIT_API IPropertyTree * getQueryById(IPropertyTree * queryRegistry, const char *queryid)
  9141. {
  9142. if (!queryRegistry || !queryid)
  9143. return NULL;
  9144. StringBuffer xpath;
  9145. xpath.append("Query[@id=\"").append(queryid).append("\"]");
  9146. return queryRegistry->getPropTree(xpath);
  9147. }
  9148. extern WORKUNIT_API IPropertyTree * getQueryById(const char *queryset, const char *queryid, bool readonly)
  9149. {
  9150. Owned<IPropertyTree> queryRegistry = getQueryRegistry(queryset, readonly);
  9151. return getQueryById(queryRegistry, queryid);
  9152. }
  9153. extern WORKUNIT_API IPropertyTree * resolveQueryAlias(IPropertyTree * queryRegistry, const char * alias)
  9154. {
  9155. if (!queryRegistry || !alias)
  9156. return NULL;
  9157. StringBuffer xpath;
  9158. unsigned cnt = 0;
  9159. StringBuffer lc(alias);
  9160. const char * search = lc.toLowerCase().str();
  9161. loop
  9162. {
  9163. xpath.set("Alias[@name='").append(search).append("']/@id");
  9164. const char * queryId = queryRegistry->queryProp(xpath);
  9165. if (!queryId)
  9166. break;
  9167. //Check for too many alias indirections.
  9168. if (cnt++ > 10)
  9169. return NULL;
  9170. search = lc.set(queryId).toLowerCase().str();
  9171. }
  9172. return getQueryById(queryRegistry, search);
  9173. }
  9174. extern WORKUNIT_API IPropertyTree * resolveQueryAlias(const char *queryset, const char *alias, bool readonly)
  9175. {
  9176. Owned<IPropertyTree> queryRegistry = getQueryRegistry(queryset, readonly);
  9177. return resolveQueryAlias(queryRegistry, alias);
  9178. }
  9179. void setQuerySuspendedState(IPropertyTree * queryRegistry, const char *id, bool suspend, const char *userid)
  9180. {
  9181. StringBuffer lcId(id);
  9182. lcId.toLowerCase();
  9183. StringBuffer xpath;
  9184. xpath.append("Query[@id=\"").append(lcId).append("\"]");
  9185. IPropertyTree *tree = queryRegistry->queryPropTree(xpath);
  9186. if (tree)
  9187. {
  9188. if (tree->getPropBool("@suspended", false) == suspend)
  9189. return;
  9190. if (suspend)
  9191. {
  9192. tree->addPropBool("@suspended", true);
  9193. if (userid && *userid)
  9194. tree->addProp("@suspendedBy", userid);
  9195. }
  9196. else
  9197. {
  9198. tree->removeProp("@suspended");
  9199. tree->removeProp("@suspendedBy");
  9200. }
  9201. }
  9202. else
  9203. throw MakeStringException((suspend)? QUERRREG_SUSPEND : QUERRREG_UNSUSPEND, "Modifying query suspended state failed. Could not find query %s", id);
  9204. }
  9205. void setQueryCommentForNamedQuery(IPropertyTree * queryRegistry, const char *id, const char *queryComment)
  9206. {
  9207. if (queryComment)
  9208. {
  9209. StringBuffer lcId(id);
  9210. lcId.toLowerCase();
  9211. StringBuffer xpath;
  9212. xpath.append("Query[@id=\"").append(lcId).append("\"]");
  9213. IPropertyTree *tree = queryRegistry->queryPropTree(xpath);
  9214. if (tree)
  9215. tree->setProp("@queryComment", queryComment);
  9216. else
  9217. throw MakeStringException(QUERRREG_COMMENT, "Could not find query %s", id);
  9218. }
  9219. }
  9220. extern WORKUNIT_API IPropertyTree * getQueryRegistryRoot()
  9221. {
  9222. Owned<IRemoteConnection> conn = querySDS().connect("/QuerySets", myProcessSession(), RTM_LOCK_READ, SDS_LOCK_TIMEOUT);
  9223. if (conn)
  9224. return conn->getRoot();
  9225. else
  9226. return NULL;
  9227. }
  9228. extern WORKUNIT_API void checkAddLibrariesToQueryEntry(IPropertyTree *queryTree, IConstWULibraryIterator *libraries)
  9229. {
  9230. if (!queryTree || !libraries)
  9231. return;
  9232. if (queryTree->hasProp("@libCount")) //already added
  9233. return;
  9234. unsigned libCount=0;
  9235. ForEach(*libraries)
  9236. {
  9237. IConstWULibrary &library = libraries->query();
  9238. SCMStringBuffer libname;
  9239. if (!library.getName(libname).length())
  9240. continue;
  9241. queryTree->addProp("Library", libname.str());
  9242. libCount++;
  9243. }
  9244. queryTree->setPropInt("@libCount", libCount);
  9245. }
  9246. extern WORKUNIT_API void checkAddLibrariesToQueryEntry(IPropertyTree *queryTree, IConstWorkUnit *cw)
  9247. {
  9248. Owned<IConstWULibraryIterator> libraries = &cw->getLibraries();
  9249. checkAddLibrariesToQueryEntry(queryTree, libraries);
  9250. }
  9251. extern WORKUNIT_API IPropertyTree * getQueryRegistry(const char * wsEclId, bool readonly)
  9252. {
  9253. //Only lock the branch for the target we're interested in.
  9254. StringBuffer xpath;
  9255. xpath.append("/QuerySets/QuerySet[@id=\"").append(wsEclId).append("\"]");
  9256. Owned<IRemoteConnection> conn = querySDS().connect(xpath.str(), myProcessSession(), readonly ? RTM_LOCK_READ : RTM_LOCK_WRITE, SDS_LOCK_TIMEOUT);
  9257. if (conn)
  9258. return conn->getRoot();
  9259. if (readonly)
  9260. return NULL;
  9261. //Lock the QuerySets in case another thread/client wants to check/add the same QuerySet.
  9262. Owned<IRemoteConnection> globalLock = querySDS().connect("/QuerySets/", myProcessSession(), RTM_LOCK_WRITE|RTM_CREATE_QUERY, SDS_LOCK_TIMEOUT);
  9263. //Re-check if the QuerySet has been added between checking the 1st time and gaining the globalLock.
  9264. conn.setown(querySDS().connect(xpath.str(), myProcessSession(), RTM_LOCK_WRITE, SDS_LOCK_TIMEOUT));
  9265. if (conn)
  9266. return conn->getRoot();
  9267. conn.setown(querySDS().connect("/QuerySets/QuerySet", myProcessSession(), RTM_LOCK_WRITE|RTM_CREATE_ADD, SDS_LOCK_TIMEOUT));
  9268. if (!conn)
  9269. throwUnexpected();
  9270. IPropertyTree * root = conn->queryRoot();
  9271. root->setProp("@id",wsEclId);
  9272. conn->commit();
  9273. return LINK(root);
  9274. }
  9275. IPropertyTree * addNamedPackageSet(IPropertyTree * packageRegistry, const char * name, IPropertyTree *packageInfo, bool overWrite)
  9276. {
  9277. StringBuffer xpath;
  9278. StringBuffer lcName(name);
  9279. lcName.toLowerCase();
  9280. // see if "name" already exists
  9281. xpath.append("Package[@id='").append(name).append("']");
  9282. IPropertyTree *pkgTree = packageRegistry->queryPropTree(xpath.str());
  9283. if (pkgTree)
  9284. {
  9285. if (overWrite)
  9286. packageRegistry->removeTree(pkgTree);
  9287. else
  9288. throw MakeStringException(WUERR_PackageAlreadyExists, "Package name %s already exists, either delete it or specify overwrite",name);
  9289. }
  9290. IPropertyTree *tree = packageRegistry->addPropTree("Package", packageInfo);
  9291. tree->setProp("@id", lcName);
  9292. return tree;
  9293. }
  9294. void removeNamedPackage(IPropertyTree * packageRegistry, const char * id)
  9295. {
  9296. StringBuffer lcId(id);
  9297. lcId.toLowerCase();
  9298. StringBuffer xpath;
  9299. xpath.append("Package[@id=\"").append(lcId).append("\"]");
  9300. packageRegistry->removeProp(xpath);
  9301. }
  9302. extern WORKUNIT_API IPropertyTree * getPackageSetRegistry(const char * wsEclId, bool readonly)
  9303. {
  9304. //Only lock the branch for the target we're interested in.
  9305. StringBuffer xpath;
  9306. xpath.append("/PackageSets/PackageSet[@id=\"").append(wsEclId).append("\"]");
  9307. Owned<IRemoteConnection> conn = querySDS().connect(xpath.str(), myProcessSession(), readonly ? RTM_LOCK_READ : RTM_LOCK_WRITE, SDS_LOCK_TIMEOUT);
  9308. if (conn)
  9309. return conn->getRoot();
  9310. if (readonly)
  9311. return NULL;
  9312. //Lock the PackageSets in case another thread/client wants to check/add the same PackageSet.
  9313. Owned<IRemoteConnection> globalLock = querySDS().connect("/PackageSets/", myProcessSession(), RTM_LOCK_WRITE|RTM_CREATE_QUERY, SDS_LOCK_TIMEOUT);
  9314. //Re-check if the PackageSet has been added between checking the 1st time and gaining the globalLock.
  9315. conn.setown(querySDS().connect(xpath.str(), myProcessSession(), RTM_LOCK_WRITE, SDS_LOCK_TIMEOUT));
  9316. if (conn)
  9317. return conn->getRoot();
  9318. conn.setown(querySDS().connect("/PackageSets/PackageSet", myProcessSession(), RTM_LOCK_WRITE|RTM_CREATE_ADD, SDS_LOCK_TIMEOUT));
  9319. if (!conn)
  9320. throwUnexpected();
  9321. IPropertyTree* root = conn->queryRoot();
  9322. root->setProp("@id",wsEclId);
  9323. conn->commit();
  9324. return LINK(root);
  9325. }
  9326. void addQueryToQuerySet(IWorkUnit *workunit, IPropertyTree *queryRegistry, const char *queryName, WUQueryActivationOptions activateOption, StringBuffer &newQueryId, const char *userid)
  9327. {
  9328. StringBuffer cleanQueryName;
  9329. appendUtf8XmlName(cleanQueryName, strlen(queryName), queryName);
  9330. SCMStringBuffer dllName;
  9331. Owned<IConstWUQuery> q = workunit->getQuery();
  9332. q->getQueryDllName(dllName);
  9333. if (!dllName.length())
  9334. throw MakeStringException(WUERR_InvalidDll, "Cannot deploy query - no associated dll.");
  9335. StringBuffer currentTargetClusterType;
  9336. queryRegistry->getProp("@targetclustertype", currentTargetClusterType);
  9337. SCMStringBuffer targetClusterType;
  9338. workunit->getDebugValue("targetclustertype", targetClusterType);
  9339. SCMStringBuffer snapshot;
  9340. workunit->getSnapshot(snapshot);
  9341. if (currentTargetClusterType.length() < 1)
  9342. {
  9343. queryRegistry->setProp("@targetclustertype", targetClusterType.str());
  9344. }
  9345. else
  9346. {
  9347. if (strcmp(currentTargetClusterType.str(), "roxie") == 0 && strcmp(currentTargetClusterType.str(), targetClusterType.str())!=0)
  9348. {
  9349. throw MakeStringException(WUERR_MismatchClusterType, "TargetClusterTypes of workunit and queryset do not match.");
  9350. }
  9351. }
  9352. IPropertyTree *newEntry = addNamedQuery(queryRegistry, cleanQueryName, workunit->queryWuid(), dllName.str(), isLibrary(workunit), userid, snapshot.str());
  9353. Owned<IConstWULibraryIterator> libraries = &workunit->getLibraries();
  9354. checkAddLibrariesToQueryEntry(newEntry, libraries);
  9355. newQueryId.append(newEntry->queryProp("@id"));
  9356. workunit->setIsQueryService(true); //will check querysets before delete
  9357. workunit->commit();
  9358. activateQuery(queryRegistry, activateOption, queryName, newQueryId, userid);
  9359. }
  9360. void activateQuery(IPropertyTree *queryRegistry, WUQueryActivationOptions activateOption, const char *queryName, const char *queryId, const char *userid)
  9361. {
  9362. StringBuffer cleanQueryName;
  9363. appendUtf8XmlName(cleanQueryName, strlen(queryName), queryName);
  9364. if (activateOption == ACTIVATE_SUSPEND_PREVIOUS|| activateOption == ACTIVATE_DELETE_PREVIOUS)
  9365. {
  9366. Owned<IPropertyTree> prevQuery = resolveQueryAlias(queryRegistry, cleanQueryName);
  9367. setQueryAlias(queryRegistry, cleanQueryName, queryId);
  9368. if (prevQuery && !streq(queryId, prevQuery->queryProp("@id")))
  9369. {
  9370. if (activateOption == ACTIVATE_SUSPEND_PREVIOUS)
  9371. setQuerySuspendedState(queryRegistry, prevQuery->queryProp("@id"), true, userid);
  9372. else
  9373. removeNamedQuery(queryRegistry, prevQuery->queryProp("@id"));
  9374. }
  9375. }
  9376. else if (activateOption == MAKE_ACTIVATE || activateOption == MAKE_ACTIVATE_LOAD_DATA_ONLY)
  9377. setQueryAlias(queryRegistry, cleanQueryName, queryId);
  9378. }
  9379. void addQueryToQuerySet(IWorkUnit *workunit, const char *querySetName, const char *queryName, WUQueryActivationOptions activateOption, StringBuffer &newQueryId, const char *userid)
  9380. {
  9381. Owned<IPropertyTree> queryRegistry = getQueryRegistry(querySetName, false);
  9382. addQueryToQuerySet(workunit, queryRegistry, queryName, activateOption, newQueryId, userid);
  9383. }
  9384. bool removeQuerySetAlias(const char *querySetName, const char *alias)
  9385. {
  9386. Owned<IPropertyTree> queryRegistry = getQueryRegistry(querySetName, true);
  9387. StringBuffer xpath;
  9388. xpath.appendf("Alias[@name='%s']", alias);
  9389. IPropertyTree *t = queryRegistry->queryPropTree(xpath);
  9390. return queryRegistry->removeTree(t);
  9391. }
  9392. void addQuerySetAlias(const char *querySetName, const char *alias, const char *id)
  9393. {
  9394. Owned<IPropertyTree> queryRegistry = getQueryRegistry(querySetName, false);
  9395. setQueryAlias(queryRegistry, alias, id);
  9396. }
  9397. void setSuspendQuerySetQuery(const char *querySetName, const char *id, bool suspend, const char *userid)
  9398. {
  9399. Owned<IPropertyTree> queryRegistry = getQueryRegistry(querySetName, true);
  9400. setQuerySuspendedState(queryRegistry, id, suspend, userid);
  9401. }
  9402. void deleteQuerySetQuery(const char *querySetName, const char *id)
  9403. {
  9404. Owned<IPropertyTree> queryRegistry = getQueryRegistry(querySetName, true);
  9405. removeNamedQuery(queryRegistry, id);
  9406. }
  9407. void removeQuerySetAliasesFromNamedQuery(const char *querySetName, const char * id)
  9408. {
  9409. Owned<IPropertyTree> queryRegistry = getQueryRegistry(querySetName, true);
  9410. clearAliases(queryRegistry, id);
  9411. }
  9412. void setQueryCommentForNamedQuery(const char *querySetName, const char *id, const char *queryComment)
  9413. {
  9414. Owned<IPropertyTree> queryRegistry = getQueryRegistry(querySetName, true);
  9415. setQueryCommentForNamedQuery(queryRegistry, id, queryComment);
  9416. }
  9417. const char *queryIdFromQuerySetWuid(IPropertyTree *queryRegistry, const char *wuid, const char *queryName, IStringVal &id)
  9418. {
  9419. if (!queryRegistry)
  9420. return NULL;
  9421. StringBuffer xpath;
  9422. xpath.appendf("Query[@wuid='%s']", wuid);
  9423. if (queryName && *queryName)
  9424. xpath.appendf("[@name='%s']", queryName);
  9425. IPropertyTree *q = queryRegistry->queryPropTree(xpath.str());
  9426. if (q)
  9427. {
  9428. id.set(q->queryProp("@id"));
  9429. }
  9430. return id.str();
  9431. }
  9432. const char *queryIdFromQuerySetWuid(const char *querySetName, const char *wuid, const char *queryName, IStringVal &id)
  9433. {
  9434. Owned<IPropertyTree> queryRegistry = getQueryRegistry(querySetName, true);
  9435. return queryIdFromQuerySetWuid(queryRegistry, wuid, queryName, id);
  9436. }
  9437. extern WORKUNIT_API void gatherLibraryNames(StringArray &names, StringArray &unresolved, IWorkUnitFactory &workunitFactory, IConstWorkUnit &cw, IPropertyTree *queryset)
  9438. {
  9439. Owned<IConstWULibraryIterator> wulibraries = &cw.getLibraries();
  9440. ForEach(*wulibraries)
  9441. {
  9442. SCMStringBuffer libname;
  9443. IConstWULibrary &wulibrary = wulibraries->query();
  9444. wulibrary.getName(libname);
  9445. if (names.contains(libname.str()) || unresolved.contains(libname.str()))
  9446. continue;
  9447. Owned<IPropertyTree> query = resolveQueryAlias(queryset, libname.str());
  9448. if (query && query->getPropBool("@isLibrary"))
  9449. {
  9450. const char *wuid = query->queryProp("@wuid");
  9451. Owned<IConstWorkUnit> libcw = workunitFactory.openWorkUnit(wuid, false);
  9452. if (libcw)
  9453. {
  9454. names.appendUniq(libname.str());
  9455. gatherLibraryNames(names, unresolved, workunitFactory, *libcw, queryset);
  9456. continue;
  9457. }
  9458. }
  9459. unresolved.appendUniq(libname.str());
  9460. }
  9461. }
  9462. bool looksLikeAWuid(const char * wuid, const char firstChar)
  9463. {
  9464. if (!wuid)
  9465. return false;
  9466. if (wuid[0] != firstChar)
  9467. return false;
  9468. if (!isdigit(wuid[1]) || !isdigit(wuid[2]) || !isdigit(wuid[3]) || !isdigit(wuid[4]))
  9469. return false;
  9470. if (!isdigit(wuid[5]) || !isdigit(wuid[6]) || !isdigit(wuid[7]) || !isdigit(wuid[8]))
  9471. return false;
  9472. return (wuid[9]=='-');
  9473. }
  9474. IPropertyTree * resolveDefinitionInArchive(IPropertyTree * archive, const char * path)
  9475. {
  9476. IPropertyTree * module = archive;
  9477. const char * dot = strrchr(path, '.');
  9478. StringBuffer xpath;
  9479. if (dot)
  9480. {
  9481. xpath.clear().append("Module[@key='").appendLower(dot-path, path).append("']");
  9482. module = archive->queryPropTree(xpath);
  9483. path = dot+1;
  9484. }
  9485. else
  9486. module = archive->queryPropTree("Module[@key='']");
  9487. if (!module)
  9488. return NULL;
  9489. xpath.clear().append("Attribute[@key='").appendLower(strlen(path), path).append("']");
  9490. return module->queryPropTree(xpath);
  9491. }
  9492. extern WORKUNIT_API void associateLocalFile(IWUQuery * query, WUFileType type, const char * name, const char * description, unsigned crc)
  9493. {
  9494. StringBuffer hostname;
  9495. queryHostIP().getIpText(hostname);
  9496. StringBuffer fullPathname;
  9497. makeAbsolutePath(name, fullPathname);
  9498. query->addAssociatedFile(type, fullPathname, hostname, description, crc);
  9499. }
  9500. extern WORKUNIT_API void descheduleWorkunit(char const * wuid)
  9501. {
  9502. Owned<IWorkUnitFactory> factory = getWorkUnitFactory();
  9503. Owned<IWorkUnit> workunit = factory->updateWorkUnit(wuid);
  9504. if(workunit)
  9505. workunit->deschedule();
  9506. else
  9507. doDescheduleWorkkunit(wuid);
  9508. }
  9509. extern WORKUNIT_API void updateWorkunitTimeStat(IWorkUnit * wu, StatisticScopeType scopeType, const char * scope, StatisticKind kind, const char * description, unsigned __int64 value)
  9510. {
  9511. wu->setStatistic(queryStatisticsComponentType(), queryStatisticsComponentName(), scopeType, scope, kind, description, value, 1, 0, StatsMergeReplace);
  9512. }
  9513. extern WORKUNIT_API void updateWorkunitTimings(IWorkUnit * wu, ITimeReporter *timer)
  9514. {
  9515. StringBuffer scope;
  9516. for (unsigned i = 0; i < timer->numSections(); i++)
  9517. {
  9518. StatisticScopeType scopeType= timer->getScopeType(i);
  9519. timer->getScope(i, scope.clear());
  9520. StatisticKind kind = timer->getTimerType(i);
  9521. wu->setStatistic(queryStatisticsComponentType(), queryStatisticsComponentName(), scopeType, scope, kind, NULL, timer->getTime(i), timer->getCount(i), timer->getMaxTime(i), StatsMergeReplace);
  9522. }
  9523. }
  9524. extern WORKUNIT_API void getWorkunitTotalTime(IConstWorkUnit* workunit, const char* creator, unsigned __int64 & totalTimeNs, unsigned __int64 & totalThisTimeNs)
  9525. {
  9526. StatisticsFilter summaryTimeFilter(SCTsummary, creator, SSTglobal, GLOBAL_SCOPE, SMeasureTimeNs, StTimeElapsed);
  9527. Owned<IConstWUStatistic> totalThorTime = getStatistic(workunit, summaryTimeFilter);
  9528. Owned<IConstWUStatistic> totalThisThorTime = workunit->getStatistic(queryStatisticsComponentName(), GLOBAL_SCOPE, StTimeElapsed);
  9529. if (totalThorTime)
  9530. totalTimeNs = totalThorTime->getValue();
  9531. else
  9532. totalTimeNs = 0;
  9533. if (totalThisThorTime)
  9534. totalThisTimeNs = totalThisThorTime->getValue();
  9535. else
  9536. totalThisTimeNs = 0;
  9537. }
  9538. extern WORKUNIT_API void addTimeStamp(IWorkUnit * wu, StatisticScopeType scopeType, const char * scope, StatisticKind kind)
  9539. {
  9540. wu->setStatistic(queryStatisticsComponentType(), queryStatisticsComponentName(), scopeType, scope, kind, NULL, getTimeStampNowValue(), 1, 0, StatsMergeAppend);
  9541. }
  9542. IConstWUStatistic * getStatistic(IConstWorkUnit * wu, const IStatisticsFilter & filter)
  9543. {
  9544. Owned<IConstWUStatisticIterator> iter = &wu->getStatistics(&filter);
  9545. if (iter->first())
  9546. return &OLINK(iter->query());
  9547. return NULL;
  9548. }
  9549. class GlobalStatisticGatherer : public CInterfaceOf<IStatisticGatherer>
  9550. {
  9551. public:
  9552. GlobalStatisticGatherer(IWorkUnit * _wu) : wu(_wu) {}
  9553. virtual void beginScope(const StatsScopeId & id)
  9554. {
  9555. prevLenStack.append(scope.length());
  9556. if (scope.length())
  9557. scope.append(":");
  9558. id.getScopeText(scope);
  9559. scopeTypeStack.append(id.queryScopeType());
  9560. }
  9561. virtual void beginSubGraphScope(unsigned id)
  9562. {
  9563. StatsScopeId scopeId(SSTsubgraph, id);
  9564. beginScope(scopeId);
  9565. }
  9566. virtual void beginActivityScope(unsigned id)
  9567. {
  9568. StatsScopeId scopeId(SSTactivity, id);
  9569. beginScope(scopeId);
  9570. }
  9571. virtual void beginEdgeScope(unsigned id, unsigned oid)
  9572. {
  9573. StatsScopeId scopeId(SSTedge, id, oid);
  9574. beginScope(scopeId);
  9575. }
  9576. virtual void endScope()
  9577. {
  9578. scope.setLength(prevLenStack.popGet());
  9579. scopeTypeStack.pop();
  9580. }
  9581. virtual void addStatistic(StatisticKind kind, unsigned __int64 value)
  9582. {
  9583. StatisticScopeType scopeType = scopeTypeStack.ordinality() ? (StatisticScopeType)scopeTypeStack.tos() : SSTglobal;
  9584. wu->setStatistic(queryStatisticsComponentType(), queryStatisticsComponentName(), scopeType, scope, kind, NULL, value, 1, 0, StatsMergeAppend);
  9585. }
  9586. virtual void updateStatistic(StatisticKind kind, unsigned __int64 value, StatsMergeAction mergeAction)
  9587. {
  9588. StatisticScopeType scopeType = scopeTypeStack.ordinality() ? (StatisticScopeType)scopeTypeStack.tos() : SSTglobal;
  9589. wu->setStatistic(queryStatisticsComponentType(), queryStatisticsComponentName(), scopeType, scope, kind, NULL, value, 1, 0, mergeAction);
  9590. }
  9591. virtual IStatisticCollection * getResult()
  9592. {
  9593. return NULL;
  9594. }
  9595. protected:
  9596. Linked<IWorkUnit> wu;
  9597. StringBuffer scope;
  9598. UnsignedArray prevLenStack;
  9599. UnsignedArray scopeTypeStack;
  9600. };
  9601. IStatisticGatherer * createGlobalStatisticGatherer(IWorkUnit * wu)
  9602. {
  9603. return new GlobalStatisticGatherer(wu);
  9604. }