fvresultset.cpp 97 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376
  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 "platform.h"
  14. #include "jliball.hpp"
  15. #include "rtlbcd.hpp"
  16. #include "workunit.hpp"
  17. #include "seclib.hpp"
  18. #include "eclrtl.hpp"
  19. #include "fvresultset.ipp"
  20. #include "fileview.hpp"
  21. #include "fverror.hpp"
  22. #include "fvdatasource.hpp"
  23. #include "fvwusource.ipp"
  24. #include "fvresultset.ipp"
  25. #include "fvdisksource.ipp"
  26. #include "fvidxsource.ipp"
  27. #include "fvrelate.ipp"
  28. #include "dasess.hpp"
  29. #include "thorxmlwrite.hpp"
  30. #include "eclhelper.hpp"
  31. #define DEFAULT_FETCH_SIZE 100
  32. #define FILEVIEW_VERSION 1
  33. #define MAX_SORT_ELEMENTS 1000
  34. //#define PAGELOADED_WORKUNITS
  35. #define MAX_FILTER_ELEMENTS 20000
  36. #define MAX_SKIP_ELEMENTS 1000
  37. #define MAX_SKIP_TIME 10000 // 10 seconds, no match then give up
  38. CResultSetMetaData * nullMeta;
  39. ITypeInfo * filePositionType;
  40. MODULE_INIT(INIT_PRIORITY_STANDARD)
  41. {
  42. nullMeta = new CResultSetMetaData(NULL, false);
  43. filePositionType = makeIntType(8, false);
  44. return true;
  45. }
  46. MODULE_EXIT()
  47. {
  48. filePositionType->Release();
  49. nullMeta->Release();
  50. }
  51. //---------------------------------------------------------------------------
  52. IFvDataSource * createDataSource(IConstWUResult * wuResult, const char * wuid, const char * username, const char * password)
  53. {
  54. Owned<ADataSource> ds;
  55. SCMStringBuffer tempFilename;
  56. wuResult->getResultFilename(tempFilename);
  57. __int64 rowLimit = wuResult->getResultRowLimit();
  58. if (tempFilename.length())
  59. ds.setown(new WorkunitDiskDataSource(tempFilename.str(), wuResult, wuid, username, password));
  60. else if (rowLimit == -2)
  61. assertex(!"Delayed queries not yet supported");
  62. else if ((rowLimit == 0) && (wuResult->getResultTotalRowCount() == 0))
  63. ds.setown(new NullDataSource);
  64. #ifdef PAGELOADED_WORKUNITS
  65. else if (wuResult->getResultDataSize() < PAGED_WU_LIMIT)
  66. ds.setown(new FullWorkUnitDataSource(wuResult, wuid));
  67. else
  68. ds.setown(new PagedWorkUnitDataSource(wuResult, wuid));
  69. #else
  70. else
  71. ds.setown(new FullWorkUnitDataSource(wuResult, wuid));
  72. #endif
  73. if (ds && ds->init())
  74. return ds.getClear();
  75. return NULL;
  76. }
  77. IFvDataSource * createFileDataSource(const char * logicalName, const char * cluster, const char * username, const char * password)
  78. {
  79. Owned<IUserDescriptor> udesc;
  80. if(username != NULL && *username != '\0')
  81. {
  82. udesc.setown(createUserDescriptor());
  83. udesc->set(username, password);
  84. }
  85. Owned<IDistributedFile> df = queryDistributedFileDirectory().lookup(logicalName, udesc.get());
  86. if (!df)
  87. throwError1(FVERR_CouldNotResolveX, logicalName);
  88. return createFileDataSource(df, logicalName, cluster, username, password);
  89. }
  90. IFvDataSource * createFileDataSource(IDistributedFile * df, const char * logicalName, const char * cluster, const char * username, const char * password)
  91. {
  92. bool blocked;
  93. if (df->isCompressed(&blocked) && !blocked)
  94. throwError1(FVERR_CompressedFile, logicalName);
  95. IPropertyTree & properties = df->queryAttributes();
  96. const char * format = properties.queryProp("@format");
  97. if (format && (stricmp(format,"csv")==0 || memicmp(format, "utf", 3) == 0))
  98. {
  99. Owned<ADataSource> ds = new DirectCsvDiskDataSource(df, format);
  100. if (ds && ds->init())
  101. return ds.getClear();
  102. return NULL;
  103. }
  104. const char * recordEcl = properties.queryProp("ECL");
  105. OwnedHqlExpr diskRecord;
  106. if (recordEcl)
  107. {
  108. diskRecord.setown(parseQuery(recordEcl));
  109. if (!diskRecord)
  110. throwError1(FVERR_BadRecordDesc, logicalName);
  111. }
  112. else
  113. {
  114. size32_t len = (size32_t)properties.getPropInt("@recordSize", 0);
  115. if (len)
  116. {
  117. VStringBuffer recordText("{ string%u contents };", len);
  118. diskRecord.setown(parseQuery(recordText));
  119. }
  120. if (!diskRecord)
  121. throwError1(FVERR_NoRecordDescription, logicalName);
  122. }
  123. Owned<ADataSource> ds;
  124. try
  125. {
  126. const char * kind = properties.queryProp("@kind");
  127. if (kind && (stricmp(kind, "key") == 0))
  128. {
  129. if (isSimplifiedRecord(diskRecord, true))
  130. ds.setown(new IndexDataSource(logicalName, diskRecord, username, password));
  131. else
  132. throwError1(FVERR_ViewComplexKey, logicalName);
  133. }
  134. else if (isSimplifiedRecord(diskRecord, false))
  135. ds.setown(new DirectDiskDataSource(logicalName, diskRecord, username, password));
  136. else if (cluster)
  137. ds.setown(new TranslatedDiskDataSource(logicalName, diskRecord, cluster, username, password));
  138. else
  139. throwError1(FVERR_NeedClusterToBrowseX, logicalName);
  140. }
  141. catch (IException * e)
  142. {
  143. ds.setown(new FailureDataSource(diskRecord, e, false, 0));
  144. e->Release();
  145. }
  146. if (ds && ds->init())
  147. return ds.getClear();
  148. return NULL;
  149. }
  150. //---------------------------------------------------------------------------
  151. static __int64 getIntBias(unsigned size)
  152. {
  153. return I64C(1) << (size * 8 - 1);
  154. }
  155. static __int64 getIntFromSwapInt(ITypeInfo & type, const void * cur, bool isMappedIndexField);
  156. static __int64 getIntFromInt(ITypeInfo & type, const void * cur, bool isMappedIndexField)
  157. {
  158. #if __BYTE_ORDER == __LITTLE_ENDIAN
  159. if (isMappedIndexField) return getIntFromSwapInt(type, cur, isMappedIndexField);
  160. #endif
  161. unsigned size=type.getSize();
  162. bool isSigned = type.isSigned();
  163. if (isSigned && !isMappedIndexField)
  164. {
  165. switch (size)
  166. {
  167. case 1: return *((signed char *)cur);
  168. case 2: return *((short *)cur);
  169. case 3: return rtlReadInt3(cur);
  170. case 4: return *((int *)cur);
  171. case 5: return rtlReadInt5(cur);
  172. case 6: return rtlReadInt6(cur);
  173. case 7: return rtlReadInt7(cur);
  174. case 8: return *((__int64 *)cur);
  175. }
  176. }
  177. else
  178. {
  179. unsigned __int64 result;
  180. switch (size)
  181. {
  182. case 1: result = *((unsigned char *)cur); break;
  183. case 2: result = *((unsigned short *)cur); break;
  184. case 3: result = rtlReadUInt3(cur); break;
  185. case 4: result = *((unsigned int *)cur); break;
  186. case 5: result = rtlReadUInt5(cur); break;
  187. case 6: result = rtlReadUInt6(cur); break;
  188. case 7: result = rtlReadUInt7(cur); break;
  189. case 8: result = *((unsigned __int64 *)cur); break;
  190. default: UNIMPLEMENTED;
  191. }
  192. if (isSigned && isMappedIndexField)
  193. result -= getIntBias(size);
  194. return result;
  195. }
  196. UNIMPLEMENTED;
  197. }
  198. static __int64 getIntFromSwapInt(ITypeInfo & type, const void * cur, bool isMappedIndexField)
  199. {
  200. #if __BYTE_ORDER != __LITTLE_ENDIAN
  201. if (insideIndex) return getIntFromInt(type, cur, isMappedIndexField);
  202. #endif
  203. unsigned size = type.getSize();
  204. bool isSigned = type.isSigned();
  205. if (isSigned && !isMappedIndexField)
  206. {
  207. switch (size)
  208. {
  209. case 1: return *((signed char *)cur);
  210. case 2: return rtlRevInt2(cur);
  211. case 3: return rtlRevInt3(cur);
  212. case 4: return rtlRevInt4(cur);
  213. case 5: return rtlRevInt5(cur);
  214. case 6: return rtlRevInt6(cur);
  215. case 7: return rtlRevInt7(cur);
  216. case 8: return rtlRevInt8(cur);
  217. }
  218. }
  219. else
  220. {
  221. unsigned __int64 result;
  222. switch (size)
  223. {
  224. case 1: result = *((unsigned char *)cur); break;
  225. case 2: result = rtlRevUInt2(cur); break;
  226. case 3: result = rtlRevUInt3(cur); break;
  227. case 4: result = rtlRevUInt4(cur); break;
  228. case 5: result = rtlRevUInt5(cur); break;
  229. case 6: result = rtlRevUInt6(cur); break;
  230. case 7: result = rtlRevUInt7(cur); break;
  231. case 8: result = rtlRevUInt8(cur); break;
  232. default:
  233. throwUnexpected();
  234. }
  235. if (isSigned && isMappedIndexField)
  236. result -= getIntBias(size);
  237. return result;
  238. }
  239. UNIMPLEMENTED;
  240. }
  241. //---------------------------------------------------------------------------
  242. CResultSetMetaData::CResultSetMetaData(IFvDataSourceMetaData * _meta, bool _useXPath)
  243. {
  244. meta = _meta;
  245. fixedSize = true;
  246. alwaysUseXPath = _useXPath;
  247. unsigned max = meta ? meta->numColumns() : 0;
  248. for (unsigned idx = 0; idx < max; idx++)
  249. {
  250. ITypeInfo * type = meta->queryType(idx);
  251. CResultSetColumnInfo * column = new CResultSetColumnInfo;
  252. column->type = type;
  253. column->flag = meta->queryFieldFlags(idx);
  254. if (type->getSize() == UNKNOWN_LENGTH)
  255. fixedSize = false;
  256. switch (type->getTypeCode())
  257. {
  258. case type_void:
  259. case type_boolean:
  260. case type_int:
  261. case type_swapint:
  262. case type_decimal:
  263. case type_real:
  264. case type_data:
  265. case type_string:
  266. case type_varstring:
  267. case type_qstring:
  268. case type_unicode:
  269. case type_varunicode:
  270. case type_utf8:
  271. case type_packedint:
  272. column->childMeta.set(nullMeta);
  273. break;
  274. case type_set:
  275. case type_table:
  276. case type_groupedtable:
  277. column->childMeta.setown(new CResultSetMetaData(meta->queryChildMeta(idx), _useXPath));
  278. break;
  279. default:
  280. UNIMPLEMENTED;
  281. }
  282. columns.append(*column);
  283. }
  284. }
  285. CResultSetMetaData::CResultSetMetaData(const CResultSetMetaData & _other)
  286. {
  287. meta = _other.meta;
  288. alwaysUseXPath = _other.alwaysUseXPath;
  289. ForEachItemIn(i, _other.columns)
  290. columns.append(OLINK(_other.columns.item(i)));
  291. fixedSize = _other.fixedSize;
  292. }
  293. void CResultSetMetaData::calcFieldOffsets(const byte * data, unsigned * offsets) const
  294. {
  295. unsigned curOffset = 0;
  296. ForEachItemIn(idx, columns)
  297. {
  298. ITypeInfo & type = *columns.item(idx).type;
  299. unsigned size = type.getSize();
  300. if (size == UNKNOWN_LENGTH)
  301. {
  302. const byte * cur = data + curOffset;
  303. switch (type.getTypeCode())
  304. {
  305. case type_data:
  306. case type_string:
  307. case type_table:
  308. case type_groupedtable:
  309. size = *((unsigned *)cur) + sizeof(unsigned);
  310. break;
  311. case type_set:
  312. size = *((unsigned *)(cur + sizeof(bool))) + sizeof(unsigned) + sizeof(bool);
  313. break;
  314. case type_qstring:
  315. size = rtlQStrSize(*((unsigned *)cur)) + sizeof(unsigned);
  316. break;
  317. case type_unicode:
  318. size = *((unsigned *)cur)*sizeof(UChar) + sizeof(unsigned);
  319. break;
  320. case type_utf8:
  321. size = sizeof(unsigned) + rtlUtf8Size(*(unsigned *)cur, cur+sizeof(unsigned));
  322. break;
  323. case type_varstring:
  324. size = strlen((char *)cur)+1;
  325. break;
  326. case type_varunicode:
  327. size = (rtlUnicodeStrlen((UChar *)cur)+1)*sizeof(UChar);
  328. break;
  329. case type_packedint:
  330. size = rtlGetPackedSize(cur);
  331. break;
  332. default:
  333. UNIMPLEMENTED;
  334. }
  335. }
  336. offsets[idx] = curOffset;
  337. curOffset += size;
  338. }
  339. offsets[columns.ordinality()] = curOffset;
  340. }
  341. IResultSetMetaData * CResultSetMetaData::getChildMeta(int column) const
  342. {
  343. if (columns.isItem(column))
  344. return LINK(columns.item(column).childMeta);
  345. return NULL;
  346. }
  347. int CResultSetMetaData::getColumnCount() const
  348. {
  349. return columns.ordinality();
  350. }
  351. DisplayType CResultSetMetaData::getColumnDisplayType(int columnIndex) const
  352. {
  353. CResultSetColumnInfo & curColumn = columns.item(columnIndex);
  354. unsigned flag = curColumn.flag;
  355. switch (flag)
  356. {
  357. case FVFFbeginif:
  358. return TypeBeginIfBlock;
  359. case FVFFendif:
  360. return TypeEndIfBlock;
  361. case FVFFbeginrecord:
  362. return TypeBeginRecord;
  363. case FVFFendrecord:
  364. return TypeEndRecord;
  365. }
  366. ITypeInfo & type = *curColumn.type;
  367. switch (type.getTypeCode())
  368. {
  369. case type_boolean:
  370. return TypeBoolean;
  371. case type_int:
  372. case type_swapint:
  373. case type_packedint:
  374. if (type.isSigned())
  375. return TypeInteger;
  376. return TypeUnsignedInteger;
  377. case type_decimal:
  378. case type_real:
  379. return TypeReal;
  380. case type_qstring:
  381. case type_string:
  382. case type_varstring:
  383. return TypeString;
  384. case type_unicode:
  385. case type_varunicode:
  386. case type_utf8:
  387. return TypeUnicode;
  388. case type_data:
  389. return TypeData;
  390. case type_set:
  391. return TypeSet;
  392. case type_table:
  393. case type_groupedtable:
  394. return TypeDataset;
  395. }
  396. UNIMPLEMENTED; // Should have been translated to one of the above by this point...
  397. return TypeUnknown;
  398. }
  399. IStringVal & CResultSetMetaData::getColumnLabel(IStringVal & s, int column) const
  400. {
  401. assertex(columns.isItem(column));
  402. s.set(meta->queryName(column));
  403. return s;
  404. }
  405. IStringVal & CResultSetMetaData::getColumnEclType(IStringVal & s, int column) const
  406. {
  407. assertex(columns.isItem(column));
  408. StringBuffer str;
  409. s.set(columns.item(column).type->getECLType(str).str());
  410. return s;
  411. }
  412. IStringVal & CResultSetMetaData::getColumnXmlType(IStringVal & s, int column) const
  413. {
  414. //This really doesn't make any sense - only makes sense to get the entire schema because of the user-defined types
  415. UNIMPLEMENTED;
  416. }
  417. bool CResultSetMetaData::isSigned(int column) const
  418. {
  419. assertex(columns.isItem(column));
  420. return columns.item(column).type->isSigned();
  421. }
  422. bool CResultSetMetaData::isEBCDIC(int column) const
  423. {
  424. assertex(columns.isItem(column));
  425. ICharsetInfo * charset = columns.item(column).type->queryCharset();
  426. return (charset && charset->queryName() == ebcdicAtom);
  427. }
  428. bool CResultSetMetaData::isBigEndian(int column) const
  429. {
  430. assertex(columns.isItem(column));
  431. ITypeInfo * type = columns.item(column).type;
  432. #if __BYTE_ORDER == __LITTLE_ENDIAN
  433. return (type->getTypeCode() == type_swapint);
  434. #else
  435. return (type->getTypeCode() != type_swapint);
  436. #endif
  437. }
  438. unsigned CResultSetMetaData::getColumnRawType(int column) const
  439. {
  440. assertex(columns.isItem(column));
  441. return getClarionResultType(columns.item(column).type);
  442. }
  443. unsigned CResultSetMetaData::getColumnRawSize(int column) const
  444. {
  445. assertex(columns.isItem(column));
  446. unsigned size = columns.item(column).type->getSize();
  447. return (size == UNKNOWN_LENGTH) ? 0 : size;
  448. }
  449. unsigned CResultSetMetaData::getNumKeyedColumns() const
  450. {
  451. return meta->numKeyedColumns();
  452. }
  453. IStringVal & CResultSetMetaData::getNaturalColumnLabel(IStringVal & s, int columnIndex) const
  454. {
  455. assertex(columns.isItem(columnIndex));
  456. CResultSetColumnInfo & column = columns.item(columnIndex);
  457. s.set(column.naturalName);
  458. return s;
  459. }
  460. bool CResultSetMetaData::isVirtual(int columnIndex) const
  461. {
  462. assertex(columns.isItem(columnIndex));
  463. CResultSetColumnInfo & column = columns.item(columnIndex);
  464. return (column.flag == FVFFvirtual);
  465. }
  466. bool CResultSetMetaData::hasGetTranslation(int columnIndex) const
  467. {
  468. assertex(columns.isItem(columnIndex));
  469. CResultSetColumnInfo & column = columns.item(columnIndex);
  470. return (column.getTransforms.ordinality() != 0);
  471. }
  472. bool CResultSetMetaData::hasSetTranslation(int columnIndex) const
  473. {
  474. assertex(columns.isItem(columnIndex));
  475. CResultSetColumnInfo & column = columns.item(columnIndex);
  476. return (column.setTransforms.ordinality() != 0);
  477. }
  478. static bool findSize(int size, IntArray &sizes)
  479. {
  480. ForEachItemIn(idx, sizes)
  481. {
  482. if (sizes.item(idx)==size)
  483. return true;
  484. }
  485. return false;
  486. }
  487. unsigned CResultSetMetaData::queryColumnIndex(unsigned firstField, const char * fieldName) const
  488. {
  489. // DonKeep track of record depth, so we don't select fields from nested records..
  490. unsigned recordDepth = 0;
  491. unsigned max = columns.ordinality();
  492. for (unsigned idx =firstField; idx < max; idx++)
  493. {
  494. CResultSetColumnInfo & column = columns.item(idx);
  495. unsigned flag = column.flag;
  496. const char * name = meta->queryName(idx);
  497. if ((recordDepth == 0) && (name && stricmp(name, fieldName) == 0))
  498. return idx;
  499. switch (flag)
  500. {
  501. case FVFFbeginrecord:
  502. recordDepth++;
  503. break;
  504. case FVFFendrecord:
  505. if (recordDepth == 0)
  506. return NotFound;
  507. recordDepth--;
  508. break;
  509. }
  510. }
  511. return NotFound;
  512. }
  513. ITypeInfo * containsSingleSimpleFieldBlankXPath(IResultSetMetaData * meta)
  514. {
  515. if (meta->getColumnCount() != 1)
  516. return NULL;
  517. CResultSetMetaData * castMeta = static_cast<CResultSetMetaData *>(meta);
  518. const char * xpath = castMeta->queryXPath(0);
  519. if (xpath && (*xpath == 0))
  520. {
  521. return castMeta->queryType(0);
  522. }
  523. return NULL;
  524. }
  525. void fvSplitXPath(const char *xpath, StringBuffer &s, const char *&name, const char **childname=NULL)
  526. {
  527. if (!xpath)
  528. return;
  529. const char * slash = strchr(xpath, '/');
  530. if (!slash)
  531. {
  532. name = xpath;
  533. if (childname)
  534. *childname = NULL;
  535. }
  536. else
  537. {
  538. if (!childname || strchr(slash+1, '/')) //output ignores xpaths that are too deep
  539. return;
  540. name = s.clear().append(slash-xpath, xpath).str();
  541. *childname = slash+1;
  542. }
  543. }
  544. void CResultSetMetaData::getXmlSchema(ISchemaBuilder & builder, bool useXPath) const
  545. {
  546. StringBuffer xname;
  547. unsigned keyedCount = getNumKeyedColumns();
  548. ForEachItemIn(idx, columns)
  549. {
  550. CResultSetColumnInfo & column = columns.item(idx);
  551. unsigned flag = column.flag;
  552. const char * name = meta->queryName(idx);
  553. const char * childname = NULL;
  554. switch (flag)
  555. {
  556. case FVFFbeginif:
  557. builder.beginIfBlock();
  558. break;
  559. case FVFFendif:
  560. builder.endIfBlock();
  561. break;
  562. case FVFFbeginrecord:
  563. if (useXPath)
  564. fvSplitXPath(meta->queryXPath(idx), xname, name);
  565. builder.beginRecord(name, meta->mixedContent(idx), NULL);
  566. break;
  567. case FVFFendrecord:
  568. if (useXPath)
  569. fvSplitXPath(meta->queryXPath(idx), xname, name);
  570. builder.endRecord(name);
  571. break;
  572. case FVFFdataset:
  573. {
  574. childname = "Row";
  575. if (useXPath)
  576. fvSplitXPath(meta->queryXPath(idx), xname, name, &childname);
  577. ITypeInfo * singleFieldType = (useXPath && name && *name && childname && *childname) ? containsSingleSimpleFieldBlankXPath(column.childMeta.get()) : NULL;
  578. if (!singleFieldType || !builder.addSingleFieldDataset(name, childname, *singleFieldType))
  579. {
  580. const CResultSetMetaData *childMeta = static_cast<const CResultSetMetaData *>(column.childMeta.get());
  581. if (builder.beginDataset(name, childname, childMeta->meta->mixedContent(), NULL))
  582. {
  583. childMeta->getXmlSchema(builder, useXPath);
  584. }
  585. builder.endDataset(name, childname);
  586. }
  587. break;
  588. }
  589. default:
  590. {
  591. ITypeInfo & type = *column.type;
  592. if (type.getTypeCode() == type_set)
  593. {
  594. childname = "Item";
  595. if (useXPath)
  596. fvSplitXPath(meta->queryXPath(idx), xname, name, &childname);
  597. builder.addSetField(name, childname, type);
  598. }
  599. else
  600. {
  601. if (useXPath)
  602. fvSplitXPath(meta->queryXPath(idx), xname, name);
  603. builder.addField(name, type, idx < keyedCount);
  604. }
  605. break;
  606. }
  607. }
  608. }
  609. }
  610. IStringVal & CResultSetMetaData::getXmlSchema(IStringVal & str, bool addHeader) const
  611. {
  612. XmlSchemaBuilder builder(addHeader);
  613. getXmlSchema(builder, alwaysUseXPath);
  614. builder.getXml(str);
  615. return str;
  616. }
  617. IStringVal & CResultSetMetaData::getXmlXPathSchema(IStringVal & str, bool addHeader) const
  618. {
  619. XmlSchemaBuilder builder(addHeader);
  620. getXmlSchema(builder, true);
  621. builder.getXml(str);
  622. return str;
  623. }
  624. //---------------------------------------------------------------------------
  625. IResultSetCursor * CResultSetBase::createCursor()
  626. {
  627. return doCreateCursor();
  628. }
  629. IResultSetCursor * CResultSetBase::createCursor(IDataVal & savedCursor)
  630. {
  631. MemoryBuffer buffer;
  632. buffer.append(savedCursor.length(), savedCursor.data());
  633. byte version;
  634. unsigned column;
  635. bool desc;
  636. buffer.read(version);
  637. buffer.read(column);
  638. if (column == (unsigned)-1)
  639. return new CResultSetCursor(getMeta(), this, buffer);
  640. buffer.read(desc);
  641. return new CResultSetSortedCursor(getMeta(), this, column, desc, buffer);
  642. }
  643. IFilteredResultSet * CResultSetBase::createFiltered()
  644. {
  645. return new CFilteredResultSetBuilder(this);
  646. }
  647. IResultSetCursor * CResultSetBase::createSortedCursor(unsigned column, bool descend)
  648. {
  649. if (getNumRows() > MAX_SORT_ELEMENTS)
  650. return NULL;
  651. return new CResultSetSortedCursor(getMeta(), this, column, descend);
  652. }
  653. CResultSetCursor * CResultSetBase::doCreateCursor()
  654. {
  655. return new CResultSetCursor(getMeta(), this);
  656. }
  657. //---------------------------------------------------------------------------
  658. CResultSet::CResultSet(IFvDataSource * _dataSource, bool _useXPath) : meta(_dataSource->queryMetaData(), _useXPath)
  659. {
  660. dataSource.set(_dataSource);
  661. if (dataSource->isIndex())
  662. calcMappedFields();
  663. }
  664. IExtendedNewResultSet * CResultSet::cloneForFilter()
  665. {
  666. Owned<IFvDataSource> clonedDataSource = dataSource->cloneForFilter();
  667. if (clonedDataSource)
  668. return new CResultSet(clonedDataSource, meta.alwaysUseXPath);
  669. return NULL;
  670. }
  671. void CResultSet::calcMappedFields()
  672. {
  673. //Work out which fields within an index record are mapped. It should be any numeric fields,
  674. //but not those within ifblocks or nested child records.... and not the fileposition
  675. unsigned max = getMetaData().getColumnCount();
  676. unsigned nesting = 0;
  677. for(unsigned i = 0; i < max; i++)
  678. {
  679. unsigned flag = meta.queryFlags(i);
  680. bool mapped = false;
  681. switch (flag)
  682. {
  683. case FVFFbeginif:
  684. case FVFFbeginrecord:
  685. nesting++;
  686. break;
  687. case FVFFendif:
  688. case FVFFendrecord:
  689. nesting--;
  690. break;
  691. default:
  692. if ((nesting == 0) && (i != max -1))
  693. {
  694. ITypeInfo * type = meta.queryType(i);
  695. switch (type->getTypeCode())
  696. {
  697. case type_int:
  698. case type_swapint:
  699. mapped = true;
  700. break;
  701. }
  702. }
  703. break;
  704. }
  705. mappedFields.append(mapped);
  706. }
  707. }
  708. int CResultSet::findColumn(const char * columnName) const
  709. {
  710. SCMStringBuffer s;
  711. for(int i = 0; i < getMetaData().getColumnCount(); i++)
  712. {
  713. s.clear();
  714. if(!stricmp(columnName, getMetaData().getColumnLabel(s, i).str()))
  715. return i;
  716. }
  717. return -1;
  718. }
  719. const IResultSetMetaData & CResultSet::getMetaData() const
  720. {
  721. return meta;
  722. }
  723. __int64 CResultSet::getNumRows() const
  724. {
  725. return dataSource->numRows();
  726. }
  727. void CResultSet::setColumnMapping(IDistributedFile * df)
  728. {
  729. StringBuffer mappingText;
  730. df->getColumnMapping(mappingText);
  731. if (mappingText.length())
  732. {
  733. FieldTransformInfoArray mappings;
  734. parseFileColumnMapping(mappings, mappingText.str(), meta);
  735. ForEachItemIn(i, mappings)
  736. {
  737. FieldTransformInfo & cur = mappings.item(i);
  738. CResultSetColumnInfo & column = meta.columns.item(cur.column);
  739. appendArray(column.getTransforms, cur.getTransforms);
  740. appendArray(column.setTransforms, cur.setTransforms);
  741. column.naturalName.setown(cur.naturalName.detach());
  742. }
  743. }
  744. }
  745. bool CResultSet::supportsRandomSeek() const
  746. {
  747. return meta.meta->supportsRandomSeek();
  748. }
  749. bool CResultSet::fetch(MemoryBuffer & out, __int64 offset)
  750. {
  751. CriticalBlock procedure(cs);
  752. return dataSource->fetchRow(out, offset);
  753. }
  754. bool CResultSet::fetchRaw(MemoryBuffer & out, __int64 offset)
  755. {
  756. CriticalBlock procedure(cs);
  757. return dataSource->fetchRawRow(out, offset);
  758. }
  759. bool CResultSet::getRow(MemoryBuffer & out, __int64 row)
  760. {
  761. CriticalBlock procedure(cs);
  762. return dataSource->getRow(out, row);
  763. }
  764. bool CResultSet::getRawRow(MemoryBuffer & out, __int64 row)
  765. {
  766. CriticalBlock procedure(cs);
  767. return dataSource->getRawRow(out, row);
  768. }
  769. bool CResultSet::isMappedIndexField(unsigned columnIndex)
  770. {
  771. return mappedFields.isItem(columnIndex) && mappedFields.item(columnIndex);
  772. }
  773. void CResultSet::serialize(MemoryBuffer & out)
  774. {
  775. out.append((byte)FILEVIEW_VERSION); // current version
  776. }
  777. //---------------------------------------------------------------------------
  778. CResultSetCursor::CResultSetCursor(const CResultSetMetaData & _meta, IExtendedNewResultSet * _resultSet) : meta(_meta)
  779. {
  780. init(_resultSet);
  781. absolute(BEFORE_FIRST_ROW);
  782. }
  783. CResultSetCursor::CResultSetCursor(const CResultSetMetaData & _meta, IExtendedNewResultSet * _resultSet, MemoryBuffer & buffer) : meta(_meta)
  784. {
  785. init(_resultSet);
  786. buffer.read(curRow);
  787. absolute(curRow);
  788. }
  789. CResultSetCursor::~CResultSetCursor()
  790. {
  791. resultSet->onClose();
  792. delete [] offsets;
  793. }
  794. void CResultSetCursor::init(IExtendedNewResultSet * _resultSet)
  795. {
  796. resultSet.set(_resultSet);
  797. offsets = new unsigned[meta.getColumnCount()+1];
  798. if (meta.isFixedSize())
  799. meta.calcFieldOffsets(NULL, offsets);
  800. resultSet->onOpen();
  801. }
  802. bool CResultSetCursor::absolute(__int64 row)
  803. {
  804. curRow = row;
  805. curRowData.clear();
  806. if ((row >= 0) && resultSet->getRow(curRowData, curRow))
  807. {
  808. if (!meta.isFixedSize())
  809. meta.calcFieldOffsets((const byte *)curRowData.toByteArray(), offsets);
  810. return true;
  811. }
  812. return false;
  813. }
  814. void CResultSetCursor::afterLast()
  815. {
  816. absolute(getNumRows()+1);
  817. curRowData.clear();
  818. }
  819. void CResultSetCursor::beforeFirst()
  820. {
  821. absolute(BEFORE_FIRST_ROW);
  822. curRowData.clear();
  823. }
  824. bool CResultSetCursor::fetch(__int64 offset)
  825. {
  826. curRow = AFTER_LAST_ROW;
  827. curRowData.clear();
  828. if (resultSet->fetch(curRowData, offset))
  829. {
  830. if (!meta.isFixedSize())
  831. meta.calcFieldOffsets((const byte *)curRowData.toByteArray(), offsets);
  832. return true;
  833. }
  834. return false;
  835. }
  836. bool CResultSetCursor::first()
  837. {
  838. return absolute(0);
  839. }
  840. static unsigned getLength(ITypeInfo & type, const byte * & cursor)
  841. {
  842. unsigned len = type.getStringLen();
  843. if (len != UNKNOWN_LENGTH)
  844. return len;
  845. len = *(unsigned *)cursor;
  846. cursor += sizeof(unsigned);
  847. return len;
  848. }
  849. bool CResultSetCursor::getBoolean(int columnIndex)
  850. {
  851. if (!isValid()) return false;
  852. const byte * cur = getColumn(columnIndex);
  853. ITypeInfo & type = *meta.columns.item(columnIndex).type;
  854. unsigned size = type.getSize();
  855. unsigned len = UNKNOWN_LENGTH; // error value
  856. switch (type.getTypeCode())
  857. {
  858. case type_void:
  859. case type_set:
  860. case type_table:
  861. case type_groupedtable:
  862. return false;
  863. case type_boolean:
  864. return *((byte *)cur) != 0;
  865. case type_int:
  866. case type_swapint:
  867. switch (size)
  868. {
  869. case 1:
  870. return *((byte *)cur) != 0;
  871. case 2:
  872. return *((short *)cur) != 0;
  873. case 4:
  874. return *((int *)cur) != 0;
  875. case 8:
  876. return *((__int64 *)cur) != 0;
  877. }
  878. break;
  879. case type_packedint:
  880. if (type.isSigned())
  881. return rtlGetPackedSigned(cur) != 0;
  882. else
  883. return rtlGetPackedUnsigned(cur) != 0;
  884. case type_decimal:
  885. if (type.isSigned())
  886. return Dec2Bool(size, cur);
  887. return UDec2Bool(size, cur);
  888. case type_real:
  889. if (size == 4)
  890. return *((float *)cur) != 0;
  891. return *((double *)cur) != 0;
  892. case type_string:
  893. len = getLength(type, cur);
  894. return rtlStrToBool(len, (const char *)cur);
  895. case type_unicode:
  896. len = getLength(type, cur);
  897. return rtlUnicodeToBool(len, (UChar const *)cur);
  898. case type_varstring:
  899. return rtlVStrToBool((const char *)cur);
  900. case type_varunicode:
  901. return rtlUnicodeToBool(rtlUnicodeStrlen((UChar const *)cur), (UChar const *)cur);
  902. case type_utf8:
  903. len = getLength(type, cur);
  904. return rtlUtf8ToBool(len, (const char *)cur);
  905. case type_qstring:
  906. len = getLength(type, cur);
  907. return rtlQStrToBool(len, (const char *)cur);
  908. case type_data:
  909. len = getLength(type, cur);
  910. return rtlDataToBool(len, cur);
  911. }
  912. UNIMPLEMENTED;
  913. return true;
  914. }
  915. IDataVal & CResultSetCursor::getBytes(IDataVal &d, int columnIndex)
  916. {
  917. if (isValid())
  918. d.setLen(getColumn(columnIndex), offsets[columnIndex+1]-offsets[columnIndex]);
  919. else
  920. d.setLen(NULL, 0);
  921. return d;
  922. }
  923. IResultSetCursor * CResultSetCursor::getChildren(int columnIndex) const
  924. {
  925. if (!isValid()) return NULL;
  926. ITypeInfo & type = *meta.columns.item(columnIndex).type;
  927. const byte * cur = getColumn(columnIndex);
  928. switch (type.getTypeCode())
  929. {
  930. case type_set:
  931. cur += sizeof(bool);
  932. break;
  933. case type_table:
  934. case type_groupedtable:
  935. break;
  936. default:
  937. return NULL;
  938. }
  939. unsigned len = *(unsigned *)cur;
  940. const byte * data = cur + sizeof(unsigned);
  941. Owned<IFvDataSource> childData = meta.meta->createChildDataSource(columnIndex, len, data);
  942. Owned<CResultSet> nestedResult = new CResultSet(childData, meta.alwaysUseXPath);
  943. return nestedResult->createCursor();
  944. }
  945. xdouble CResultSetCursor::getDouble(int columnIndex)
  946. {
  947. if (!isValid()) return 0.0;
  948. const byte * cur = getColumn(columnIndex);
  949. ITypeInfo & type = *meta.columns.item(columnIndex).type;
  950. unsigned size = type.getSize();
  951. unsigned len = UNKNOWN_LENGTH; // error value
  952. switch (type.getTypeCode())
  953. {
  954. case type_void:
  955. case type_set:
  956. case type_table:
  957. case type_groupedtable:
  958. return 0;
  959. case type_boolean:
  960. return *((byte *)cur) != 0;
  961. case type_int:
  962. if (type.isSigned())
  963. return (xdouble)getIntFromInt(type, cur, isMappedIndexField(columnIndex));
  964. return (xdouble)(__int64)getIntFromInt(type, cur, isMappedIndexField(columnIndex));
  965. case type_swapint:
  966. if (type.isSigned())
  967. return (xdouble)getIntFromSwapInt(type, cur, isMappedIndexField(columnIndex));
  968. return (xdouble)(__int64)getIntFromSwapInt(type, cur, isMappedIndexField(columnIndex));
  969. case type_packedint:
  970. if (type.isSigned())
  971. return (xdouble)rtlGetPackedSigned(cur);
  972. else
  973. return (xdouble)rtlGetPackedUnsigned(cur);
  974. case type_decimal:
  975. {
  976. DecLock();
  977. if (type.isSigned())
  978. DecPushDecimal(cur, type.getSize(), type.getPrecision());
  979. else
  980. DecPushUDecimal(cur, type.getSize(), type.getPrecision());
  981. xdouble ret = DecPopReal();
  982. DecUnlock();
  983. return ret;
  984. }
  985. case type_real:
  986. if (size == 4)
  987. return *((float *)cur);
  988. return *((double *)cur);
  989. case type_data:
  990. case type_string:
  991. len = getLength(type, cur);
  992. return rtlStrToReal(len, (const char *)cur);
  993. case type_qstring:
  994. {
  995. len = getLength(type, cur);
  996. unsigned newSize;
  997. char * newStr;
  998. rtlQStrToStrX(newSize, newStr, len, (const char *)cur);
  999. double ret = rtlStrToReal(newSize, newStr);
  1000. rtlFree(newStr);
  1001. return ret;
  1002. }
  1003. case type_unicode:
  1004. len = getLength(type, cur);
  1005. return rtlUnicodeToReal(len, (UChar const *)cur);
  1006. case type_utf8:
  1007. len = getLength(type, cur);
  1008. return rtlUtf8ToReal(len, (const char *)cur);
  1009. case type_varstring:
  1010. return rtlVStrToReal((const char *)cur);
  1011. case type_varunicode:
  1012. return rtlUnicodeToReal(rtlUnicodeStrlen((UChar const *)cur), (UChar const *)cur);
  1013. }
  1014. UNIMPLEMENTED;
  1015. return 0.0;
  1016. }
  1017. int CResultSetCursor::getFetchSize() const
  1018. {
  1019. return DEFAULT_FETCH_SIZE;
  1020. }
  1021. __int64 CResultSetCursor::getInt(int columnIndex)
  1022. {
  1023. if (!isValid()) return 0;
  1024. const byte * cur = getColumn(columnIndex);
  1025. ITypeInfo & type = *meta.columns.item(columnIndex).type;
  1026. unsigned size = type.getSize();
  1027. unsigned len = UNKNOWN_LENGTH;
  1028. switch (type.getTypeCode())
  1029. {
  1030. case type_void:
  1031. case type_set:
  1032. case type_table:
  1033. case type_groupedtable:
  1034. return 0;
  1035. case type_boolean:
  1036. return *((byte *)cur) != 0;
  1037. case type_int:
  1038. return getIntFromInt(type, cur, isMappedIndexField(columnIndex));
  1039. case type_swapint:
  1040. return getIntFromSwapInt(type, cur, isMappedIndexField(columnIndex));
  1041. case type_packedint:
  1042. if (type.isSigned())
  1043. return rtlGetPackedSigned(cur);
  1044. else
  1045. return rtlGetPackedUnsigned(cur);
  1046. case type_decimal:
  1047. {
  1048. DecLock();
  1049. if (type.isSigned())
  1050. DecPushDecimal(cur, type.getSize(), type.getPrecision());
  1051. else
  1052. DecPushUDecimal(cur, type.getSize(), type.getPrecision());
  1053. __int64 ret = DecPopInt64();
  1054. DecUnlock();
  1055. return ret;
  1056. }
  1057. case type_real:
  1058. if (size == 4)
  1059. return (__int64) *((float *)cur);
  1060. return (__int64) *((double *)cur);
  1061. case type_data:
  1062. case type_string:
  1063. len = getLength(type, cur);
  1064. return rtlStrToInt8(len, (const char *)cur);
  1065. case type_qstring:
  1066. {
  1067. unsigned newSize;
  1068. char * newStr;
  1069. len = getLength(type, cur);
  1070. rtlQStrToStrX(newSize, newStr, len, (const char *)cur);
  1071. __int64 ret = rtlStrToInt8(newSize, newStr);
  1072. rtlFree(newStr);
  1073. return ret;
  1074. }
  1075. case type_unicode:
  1076. len = getLength(type, cur);
  1077. return rtlUnicodeToInt8(len, (UChar const *)cur);
  1078. case type_utf8:
  1079. len = getLength(type, cur);
  1080. return rtlUtf8ToInt(len, (const char *)cur);
  1081. case type_varstring:
  1082. return rtlVStrToInt8((const char *)cur);
  1083. case type_varunicode:
  1084. return rtlUnicodeToInt8(rtlUnicodeStrlen((UChar const *)cur), (UChar const *)cur);
  1085. }
  1086. UNIMPLEMENTED;
  1087. return 0;
  1088. }
  1089. bool CResultSetCursor::getIsAll(int columnIndex) const
  1090. {
  1091. if (!isValid()) return false;
  1092. ITypeInfo & type = *meta.columns.item(columnIndex).type;
  1093. if (type.getTypeCode() != type_set)
  1094. return false;
  1095. const byte * cur = getColumn(columnIndex);
  1096. return *(bool *)cur;
  1097. }
  1098. int CResultSetCursor::getType()
  1099. {
  1100. if (getNumRows() != UNKNOWN_NUM_ROWS)
  1101. return TYPE_SCROLL_INSENSITIVE;
  1102. return TYPE_FORWARD_ONLY;
  1103. }
  1104. const IResultSetMetaData & CResultSetCursor::getMetaData() const
  1105. {
  1106. return meta;
  1107. }
  1108. __int64 CResultSetCursor::getCurRow() const
  1109. {
  1110. return curRow;
  1111. }
  1112. __int64 CResultSetCursor::getNumRows() const
  1113. {
  1114. return resultSet->getNumRows();
  1115. }
  1116. IDataVal & CResultSetCursor::getRaw(IDataVal &d, int columnIndex)
  1117. {
  1118. //MORE: This should work on the raw data!
  1119. return getBytes(d, columnIndex);
  1120. }
  1121. IDataVal & CResultSetCursor::getRawRow(IDataVal &d)
  1122. {
  1123. MemoryBuffer temp;
  1124. if (resultSet->getRawRow(temp, curRow))
  1125. d.setLen(temp.toByteArray(), temp.length());
  1126. return d;
  1127. }
  1128. __int64 CResultSetCursor::translateRow(__int64 row) const
  1129. {
  1130. return row;
  1131. }
  1132. IStringVal & CResultSetCursor::getString(IStringVal & ret, int columnIndex)
  1133. {
  1134. if (!isValid())
  1135. {
  1136. ret.set("");
  1137. return ret;
  1138. }
  1139. const byte * cur = getColumn(columnIndex);
  1140. unsigned resultLen;
  1141. char * resultStr = NULL;
  1142. ITypeInfo & type = *meta.columns.item(columnIndex).type;
  1143. unsigned size = type.getSize();
  1144. unsigned len;
  1145. switch (type.getTypeCode())
  1146. {
  1147. case type_void:
  1148. case type_set:
  1149. case type_table:
  1150. case type_groupedtable:
  1151. ret.set("");
  1152. break;
  1153. case type_boolean:
  1154. if (*((byte *)cur) != 0)
  1155. ret.set("1");
  1156. else
  1157. ret.set("");
  1158. break;
  1159. case type_int:
  1160. {
  1161. __int64 value = getIntFromInt(type, cur, isMappedIndexField(columnIndex));
  1162. if (type.isSigned())
  1163. rtlInt8ToStrX(resultLen, resultStr, value);
  1164. else
  1165. rtlUInt8ToStrX(resultLen, resultStr, (unsigned __int64) value);
  1166. ret.setLen(resultStr, resultLen);
  1167. break;
  1168. }
  1169. case type_swapint:
  1170. {
  1171. __int64 value = getIntFromSwapInt(type, cur, isMappedIndexField(columnIndex));
  1172. if (type.isSigned())
  1173. rtlInt8ToStrX(resultLen, resultStr, value);
  1174. else
  1175. rtlUInt8ToStrX(resultLen, resultStr, (unsigned __int64) value);
  1176. ret.setLen(resultStr, resultLen);
  1177. break;
  1178. }
  1179. case type_packedint:
  1180. {
  1181. if (type.isSigned())
  1182. rtlInt8ToStrX(resultLen, resultStr, rtlGetPackedSigned(cur));
  1183. else
  1184. rtlUInt8ToStrX(resultLen, resultStr, rtlGetPackedUnsigned(cur));
  1185. ret.setLen(resultStr, resultLen);
  1186. break;
  1187. }
  1188. case type_decimal:
  1189. {
  1190. DecLock();
  1191. if (type.isSigned())
  1192. DecPushDecimal(cur, type.getSize(), type.getPrecision());
  1193. else
  1194. DecPushUDecimal(cur, type.getSize(), type.getPrecision());
  1195. DecPopStringX(resultLen, resultStr);
  1196. DecUnlock();
  1197. ret.setLen(resultStr, resultLen);
  1198. return ret;
  1199. }
  1200. case type_real:
  1201. if (size == 4)
  1202. rtlRealToStrX(resultLen, resultStr, *(float *)cur);
  1203. else
  1204. rtlRealToStrX(resultLen, resultStr, *(double *)cur);
  1205. ret.setLen(resultStr, resultLen);
  1206. break;
  1207. case type_qstring:
  1208. len = getLength(type, cur);
  1209. rtlQStrToStrX(resultLen, resultStr, len, (const char *)cur);
  1210. ret.setLen(resultStr, resultLen);
  1211. break;
  1212. case type_data:
  1213. case type_string:
  1214. len = getLength(type, cur);
  1215. ret.setLen((const char *)cur, len);
  1216. break;
  1217. case type_unicode:
  1218. len = getLength(type, cur);
  1219. rtlUnicodeToStrX(resultLen, resultStr, len, (UChar const *)cur);
  1220. ret.setLen(resultStr, resultLen);
  1221. break;
  1222. case type_utf8:
  1223. len = getLength(type, cur);
  1224. rtlUtf8ToStrX(resultLen, resultStr, len, (const char *)cur);
  1225. ret.setLen(resultStr, resultLen);
  1226. break;
  1227. case type_varstring:
  1228. ret.set((const char *)cur);
  1229. break;
  1230. case type_varunicode:
  1231. rtlUnicodeToStrX(resultLen, resultStr, rtlUnicodeStrlen((UChar const *)cur), (UChar const *)cur);
  1232. ret.setLen(resultStr, resultLen);
  1233. break;
  1234. default:
  1235. UNIMPLEMENTED;
  1236. }
  1237. free(resultStr);
  1238. return ret;
  1239. }
  1240. IStringVal & CResultSetCursor::getDisplayText(IStringVal &ret, int columnIndex)
  1241. {
  1242. if (!isValid())
  1243. {
  1244. ret.set("");
  1245. return ret;
  1246. }
  1247. CResultSetColumnInfo & column = meta.columns.item(columnIndex);
  1248. unsigned flags = column.flag;
  1249. switch (flags)
  1250. {
  1251. case FVFFbeginif:
  1252. case FVFFendif:
  1253. case FVFFbeginrecord:
  1254. case FVFFendrecord:
  1255. case FVFFdataset:
  1256. case FVFFset:
  1257. ret.set("");
  1258. return ret;
  1259. }
  1260. const byte * cur = getColumn(columnIndex);
  1261. unsigned resultLen;
  1262. char * resultStr = NULL;
  1263. ITypeInfo & type = *column.type;
  1264. unsigned size = type.getSize();
  1265. unsigned len = UNKNOWN_LENGTH;
  1266. switch (type.getTypeCode())
  1267. {
  1268. case type_boolean:
  1269. if (*((byte *)cur) != 0)
  1270. ret.set("true");
  1271. else
  1272. ret.set("false");
  1273. break;
  1274. case type_int:
  1275. {
  1276. __int64 value = getIntFromInt(type, cur, isMappedIndexField(columnIndex));
  1277. if (type.isSigned())
  1278. rtlInt8ToStrX(resultLen, resultStr, value);
  1279. else
  1280. rtlUInt8ToStrX(resultLen, resultStr, (unsigned __int64) value);
  1281. ret.setLen(resultStr, resultLen);
  1282. break;
  1283. }
  1284. case type_swapint:
  1285. {
  1286. __int64 value = getIntFromSwapInt(type, cur, isMappedIndexField(columnIndex));
  1287. if (type.isSigned())
  1288. rtlInt8ToStrX(resultLen, resultStr, value);
  1289. else
  1290. rtlUInt8ToStrX(resultLen, resultStr, (unsigned __int64) value);
  1291. ret.setLen(resultStr, resultLen);
  1292. break;
  1293. }
  1294. case type_packedint:
  1295. {
  1296. if (type.isSigned())
  1297. rtlInt8ToStrX(resultLen, resultStr, rtlGetPackedSigned(cur));
  1298. else
  1299. rtlUInt8ToStrX(resultLen, resultStr, rtlGetPackedUnsigned(cur));
  1300. ret.setLen(resultStr, resultLen);
  1301. break;
  1302. }
  1303. case type_decimal:
  1304. {
  1305. DecLock();
  1306. if (type.isSigned())
  1307. DecPushDecimal(cur, type.getSize(), type.getPrecision());
  1308. else
  1309. DecPushUDecimal(cur, type.getSize(), type.getPrecision());
  1310. DecPopStringX(resultLen, resultStr);
  1311. DecUnlock();
  1312. ret.setLen(resultStr, resultLen);
  1313. return ret;
  1314. }
  1315. case type_real:
  1316. if (size == 4)
  1317. rtlRealToStrX(resultLen, resultStr, *(float *)cur);
  1318. else
  1319. rtlRealToStrX(resultLen, resultStr, *(double *)cur);
  1320. ret.setLen(resultStr, resultLen);
  1321. break;
  1322. case type_qstring:
  1323. len = getLength(type, cur);
  1324. rtlQStrToStrX(resultLen, resultStr, len, (const char *)cur);
  1325. ret.setLen(resultStr, resultLen);
  1326. break;
  1327. case type_data:
  1328. {
  1329. len = getLength(type, cur);
  1330. StringBuffer temp;
  1331. while (len--)
  1332. temp.appendhex(*cur++, true);
  1333. ret.setLen(temp.str(), temp.length());
  1334. break;
  1335. }
  1336. case type_string:
  1337. {
  1338. len = getLength(type, cur);
  1339. rtlStrToUtf8X(resultLen, resultStr, len , (const char *)cur);
  1340. ret.setLen(resultStr, rtlUtf8Size(resultLen, resultStr));
  1341. break;
  1342. }
  1343. case type_unicode:
  1344. len = getLength(type, cur);
  1345. rtlUnicodeToUtf8X(resultLen, resultStr, len, (UChar const *)cur);
  1346. ret.setLen(resultStr, rtlUtf8Size(resultLen, resultStr));
  1347. break;
  1348. case type_utf8:
  1349. len = getLength(type, cur);
  1350. ret.setLen((const char *)cur, rtlUtf8Size(len, cur));
  1351. break;
  1352. case type_varstring:
  1353. ret.set((const char *)cur);
  1354. break;
  1355. case type_varunicode:
  1356. rtlUnicodeToCodepageX(resultLen, resultStr, rtlUnicodeStrlen((UChar const *)cur), (UChar const *)cur, "UTF-8");
  1357. ret.setLen(resultStr, resultLen);
  1358. break;
  1359. default:
  1360. UNIMPLEMENTED;
  1361. }
  1362. rtlFree(resultStr);
  1363. return ret;
  1364. }
  1365. void CResultSetCursor::writeXmlText(IXmlWriter &writer, int columnIndex, const char *tag)
  1366. {
  1367. if (!isValid())
  1368. return;
  1369. const char * name = (tag) ? tag : meta.meta->queryXmlTag(columnIndex);
  1370. CResultSetColumnInfo & column = meta.columns.item(columnIndex);
  1371. unsigned flags = column.flag;
  1372. switch (flags)
  1373. {
  1374. case FVFFbeginif:
  1375. case FVFFendif:
  1376. return;
  1377. case FVFFbeginrecord:
  1378. {
  1379. if (name && *name)
  1380. {
  1381. writer.outputBeginNested(name, false);
  1382. const IntArray &attributes = meta.meta->queryAttrList(columnIndex);
  1383. ForEachItemIn(ac, attributes)
  1384. writeXmlText(writer, attributes.item(ac), NULL);
  1385. }
  1386. }
  1387. return;
  1388. case FVFFendrecord:
  1389. if (name && *name)
  1390. writer.outputEndNested(name);
  1391. return;
  1392. }
  1393. const byte * cur = getColumn(columnIndex);
  1394. unsigned resultLen;
  1395. char * resultStr = NULL;
  1396. ITypeInfo & type = *column.type;
  1397. unsigned size = type.getSize();
  1398. unsigned len = UNKNOWN_LENGTH;
  1399. switch (type.getTypeCode())
  1400. {
  1401. case type_boolean:
  1402. writer.outputBool(*((byte *)cur) != 0, name);
  1403. break;
  1404. case type_int:
  1405. {
  1406. __int64 value = getIntFromInt(type, cur, isMappedIndexField(columnIndex));
  1407. if (type.isSigned())
  1408. writer.outputInt((__int64) value, name);
  1409. else
  1410. writer.outputUInt((unsigned __int64) value, name);
  1411. break;
  1412. }
  1413. case type_swapint:
  1414. {
  1415. __int64 value = getIntFromSwapInt(type, cur, isMappedIndexField(columnIndex));
  1416. if (type.isSigned())
  1417. writer.outputInt((__int64) value, name);
  1418. else
  1419. writer.outputUInt((unsigned __int64) value, name);
  1420. break;
  1421. }
  1422. case type_packedint:
  1423. {
  1424. if (type.isSigned())
  1425. writer.outputInt(rtlGetPackedSigned(cur), name);
  1426. else
  1427. writer.outputUInt(rtlGetPackedUnsigned(cur), name);
  1428. break;
  1429. }
  1430. case type_decimal:
  1431. if (type.isSigned())
  1432. writer.outputDecimal(cur, size, type.getPrecision(), name);
  1433. else
  1434. writer.outputUDecimal(cur, size, type.getPrecision(), name);
  1435. break;
  1436. case type_real:
  1437. if (size == 4)
  1438. writer.outputReal(*(float *)cur, name);
  1439. else
  1440. writer.outputReal(*(double *)cur, name);
  1441. break;
  1442. case type_qstring:
  1443. len = getLength(type, cur);
  1444. rtlQStrToStrX(resultLen, resultStr, len, (const char *)cur);
  1445. writer.outputString(resultLen, resultStr, name);
  1446. break;
  1447. case type_data:
  1448. len = getLength(type, cur);
  1449. writer.outputData(len, cur, name);
  1450. break;
  1451. case type_string:
  1452. len = getLength(type, cur);
  1453. if (meta.isEBCDIC(columnIndex))
  1454. {
  1455. rtlEStrToStrX(resultLen, resultStr, len, (const char *)cur);
  1456. writer.outputString(resultLen, resultStr, name);
  1457. }
  1458. else
  1459. writer.outputString(len, (const char *)cur, name);
  1460. break;
  1461. case type_unicode:
  1462. len = getLength(type, cur);
  1463. writer.outputUnicode(len, (UChar const *)cur, name);
  1464. break;
  1465. case type_varstring:
  1466. if (meta.isEBCDIC(columnIndex))
  1467. {
  1468. rtlStrToEStrX(resultLen, resultStr, strlen((const char *)cur), (const char *)cur);
  1469. writer.outputString(resultLen, resultStr, name);
  1470. }
  1471. else
  1472. writer.outputString(strlen((const char *)cur), (const char *)cur, name);
  1473. break;
  1474. case type_varunicode:
  1475. writer.outputUnicode(rtlUnicodeStrlen((UChar const *)cur), (UChar const *)cur, name);
  1476. break;
  1477. case type_utf8:
  1478. len = getLength(type, cur);
  1479. writer.outputUtf8(len, (const char *)cur, name);
  1480. break;
  1481. case type_table:
  1482. case type_groupedtable:
  1483. {
  1484. writer.outputBeginNested(name, false);
  1485. Owned<IResultSetCursor> childCursor = getChildren(columnIndex);
  1486. childCursor->beginWriteXmlRows(writer);
  1487. ForEach(*childCursor)
  1488. childCursor->writeXmlRow(writer);
  1489. childCursor->endWriteXmlRows(writer);
  1490. writer.outputEndNested(name);
  1491. }
  1492. break;
  1493. case type_set:
  1494. {
  1495. writer.outputBeginNested(name, false);
  1496. if (getIsAll(columnIndex))
  1497. writer.outputSetAll();
  1498. else
  1499. {
  1500. Owned<IResultSetCursor> childCursor = getChildren(columnIndex);
  1501. childCursor->beginWriteXmlRows(writer);
  1502. ForEach(*childCursor)
  1503. childCursor->writeXmlItem(writer);
  1504. childCursor->endWriteXmlRows(writer);
  1505. }
  1506. writer.outputEndNested(name);
  1507. }
  1508. break;
  1509. default:
  1510. UNIMPLEMENTED;
  1511. }
  1512. rtlFree(resultStr);
  1513. }
  1514. IStringVal & CResultSetCursor::getXml(IStringVal &ret, int columnIndex)
  1515. {
  1516. Owned<CommonXmlWriter> writer = CreateCommonXmlWriter(XWFexpandempty);
  1517. writeXmlText(*writer, columnIndex);
  1518. ret.set(writer->str());
  1519. return ret;
  1520. }
  1521. IStringVal & CResultSetCursor::getXmlItem(IStringVal & ret)
  1522. {
  1523. Owned<CommonXmlWriter> writer = CreateCommonXmlWriter(XWFexpandempty);
  1524. writeXmlText(*writer, 0, meta.meta->queryXmlTag());
  1525. ret.set(writer->str());
  1526. return ret;
  1527. }
  1528. void CResultSetCursor::writeXmlItem(IXmlWriter &writer)
  1529. {
  1530. writeXmlText(writer, 0, meta.meta->queryXmlTag());
  1531. }
  1532. IStringVal & CResultSetCursor::getXmlRow(IStringVal &ret)
  1533. {
  1534. Owned<CommonXmlWriter> writer = CreateCommonXmlWriter(XWFexpandempty);
  1535. writeXmlRow(*writer);
  1536. ret.set(writer->str());
  1537. return ret;
  1538. }
  1539. void CResultSetCursor::beginWriteXmlRows(IXmlWriter & writer)
  1540. {
  1541. const char *rowtag = meta.meta->queryXmlTag();
  1542. if (rowtag && *rowtag)
  1543. writer.outputBeginArray(rowtag);
  1544. }
  1545. void CResultSetCursor::endWriteXmlRows(IXmlWriter & writer)
  1546. {
  1547. const char *rowtag = meta.meta->queryXmlTag();
  1548. if (rowtag && *rowtag)
  1549. writer.outputEndArray(rowtag);
  1550. }
  1551. void CResultSetCursor::writeXmlRow(IXmlWriter &writer)
  1552. {
  1553. StringBuffer temp;
  1554. const char *rowtag = meta.meta->queryXmlTag();
  1555. if (rowtag && *rowtag)
  1556. {
  1557. writer.outputBeginNested(rowtag, false);
  1558. const IntArray &attributes = meta.meta->queryAttrList();
  1559. ForEachItemIn(ac, attributes)
  1560. writeXmlText(writer, attributes.item(ac), NULL);
  1561. }
  1562. unsigned numColumns = meta.getColumnCount();
  1563. unsigned ignoreNesting = 0;
  1564. for (unsigned col = 0; col < numColumns; col++)
  1565. {
  1566. unsigned flags = meta.columns.item(col).flag;
  1567. const char *tag = meta.meta->queryXmlTag(col);
  1568. if (tag && *tag=='@')
  1569. continue;
  1570. switch (flags)
  1571. {
  1572. case FVFFbeginif:
  1573. if (ignoreNesting || !getBoolean(col))
  1574. ignoreNesting++;
  1575. break;
  1576. case FVFFendif:
  1577. if (ignoreNesting)
  1578. ignoreNesting--;
  1579. break;
  1580. case FVFFbeginrecord:
  1581. if (ignoreNesting)
  1582. ignoreNesting++;
  1583. else
  1584. writeXmlText(writer, col);
  1585. break;
  1586. case FVFFendrecord:
  1587. if (ignoreNesting)
  1588. ignoreNesting--;
  1589. else
  1590. writeXmlText(writer, col);
  1591. break;
  1592. case FVFFnone:
  1593. case FVFFvirtual:
  1594. case FVFFdataset:
  1595. case FVFFset:
  1596. if (ignoreNesting == 0)
  1597. writeXmlText(writer, col);
  1598. break;
  1599. }
  1600. }
  1601. assertex(ignoreNesting == 0);
  1602. writer.outputEndNested(rowtag);
  1603. }
  1604. bool CResultSetCursor::isAfterLast() const
  1605. {
  1606. return (curRowData.length() == 0) && (getCurRow() != BEFORE_FIRST_ROW);
  1607. }
  1608. bool CResultSetCursor::isBeforeFirst() const
  1609. {
  1610. return getCurRow() == BEFORE_FIRST_ROW;
  1611. }
  1612. bool CResultSetCursor::isFirst() const
  1613. {
  1614. return (getCurRow() == 0);
  1615. }
  1616. bool CResultSetCursor::isLast() const
  1617. {
  1618. if (curRowData.length() == 0)
  1619. return false;
  1620. __int64 numRows = getNumRows();
  1621. if (numRows != UNKNOWN_NUM_ROWS)
  1622. return getCurRow() == numRows+1;
  1623. MemoryBuffer temp;
  1624. return !resultSet->getRow(temp, translateRow(getCurRow()+1));
  1625. }
  1626. bool CResultSetCursor::isValid() const
  1627. {
  1628. return (curRowData.length() != 0);
  1629. }
  1630. bool CResultSetCursor::isNull(int columnIndex) const
  1631. {
  1632. //MORE: There needs to be some projected extra field to
  1633. return false;
  1634. }
  1635. bool CResultSetCursor::last()
  1636. {
  1637. return absolute(getNumRows()-1);
  1638. }
  1639. bool CResultSetCursor::next()
  1640. {
  1641. return absolute(getCurRow()+1);
  1642. }
  1643. bool CResultSetCursor::previous()
  1644. {
  1645. return absolute(getCurRow()-1);
  1646. }
  1647. bool CResultSetCursor::relative(__int64 rows)
  1648. {
  1649. if (getCurRow() + rows < 0)
  1650. {
  1651. beforeFirst();
  1652. return false;
  1653. }
  1654. else
  1655. return absolute (getCurRow() + rows);
  1656. }
  1657. void CResultSetCursor::serialize(IDataVal & d)
  1658. {
  1659. MemoryBuffer buffer;
  1660. resultSet->serialize(buffer);
  1661. serializeType(buffer);
  1662. serialize(buffer);
  1663. d.setLen(buffer.toByteArray(), buffer.length());
  1664. }
  1665. void CResultSetCursor::serializeType(MemoryBuffer & buffer)
  1666. {
  1667. buffer.append((unsigned)-1);
  1668. }
  1669. void CResultSetCursor::serialize(MemoryBuffer & buffer)
  1670. {
  1671. buffer.append(getCurRow());
  1672. }
  1673. void CResultSetCursor::setFetchSize(int rows)
  1674. {
  1675. }
  1676. bool CResultSetCursor::supportsRandomSeek() const
  1677. {
  1678. return meta.meta->supportsRandomSeek();
  1679. }
  1680. //---------------------------------------------------------------------------
  1681. bool IndirectResultSetCursor::absolute(__int64 row)
  1682. {
  1683. return queryBase()->absolute(row);
  1684. }
  1685. void IndirectResultSetCursor::afterLast()
  1686. {
  1687. queryBase()->afterLast();
  1688. }
  1689. void IndirectResultSetCursor::beforeFirst()
  1690. {
  1691. queryBase()->beforeFirst();
  1692. }
  1693. bool IndirectResultSetCursor::fetch(__int64 fileoffset)
  1694. {
  1695. return queryBase()->fetch(fileoffset);
  1696. }
  1697. bool IndirectResultSetCursor::first()
  1698. {
  1699. return queryBase()->first();
  1700. }
  1701. bool IndirectResultSetCursor::getBoolean(int columnIndex)
  1702. {
  1703. return queryBase()->getBoolean(columnIndex);
  1704. }
  1705. IDataVal & IndirectResultSetCursor::getBytes(IDataVal &d, int columnIndex)
  1706. {
  1707. return queryBase()->getBytes(d, columnIndex);
  1708. }
  1709. IResultSetCursor * IndirectResultSetCursor::getChildren(int columnIndex) const
  1710. {
  1711. return queryBase()->getChildren(columnIndex);
  1712. }
  1713. xdouble IndirectResultSetCursor::getDouble(int columnIndex)
  1714. {
  1715. return queryBase()->getDouble(columnIndex);
  1716. }
  1717. int IndirectResultSetCursor::getFetchSize() const
  1718. {
  1719. return queryBase()->getFetchSize();
  1720. }
  1721. bool IndirectResultSetCursor::getIsAll(int columnIndex) const
  1722. {
  1723. return queryBase()->getIsAll(columnIndex);
  1724. }
  1725. __int64 IndirectResultSetCursor::getInt(int columnIndex)
  1726. {
  1727. return queryBase()->getInt(columnIndex);
  1728. }
  1729. IDataVal & IndirectResultSetCursor::getRaw(IDataVal &d, int columnIndex)
  1730. {
  1731. return queryBase()->getRaw(d, columnIndex);
  1732. }
  1733. IDataVal & IndirectResultSetCursor::getRawRow(IDataVal &d)
  1734. {
  1735. return queryBase()->getRawRow(d);
  1736. }
  1737. __int64 IndirectResultSetCursor::getNumRows() const
  1738. {
  1739. return queryBase()->getNumRows();
  1740. }
  1741. IStringVal & IndirectResultSetCursor::getString(IStringVal & ret, int columnIndex)
  1742. {
  1743. return queryBase()->getString(ret, columnIndex);
  1744. }
  1745. bool IndirectResultSetCursor::isAfterLast() const
  1746. {
  1747. return queryBase()->isAfterLast();
  1748. }
  1749. bool IndirectResultSetCursor::isBeforeFirst() const
  1750. {
  1751. return queryBase()->isBeforeFirst();
  1752. }
  1753. bool IndirectResultSetCursor::isFirst() const
  1754. {
  1755. return queryBase()->isFirst();
  1756. }
  1757. bool IndirectResultSetCursor::isLast() const
  1758. {
  1759. return queryBase()->isLast();
  1760. }
  1761. bool IndirectResultSetCursor::isNull(int columnIndex) const
  1762. {
  1763. return queryBase()->isNull(columnIndex);
  1764. }
  1765. bool IndirectResultSetCursor::isValid() const
  1766. {
  1767. return queryBase()->isValid();
  1768. }
  1769. bool IndirectResultSetCursor::last()
  1770. {
  1771. return queryBase()->last();
  1772. }
  1773. bool IndirectResultSetCursor::next()
  1774. {
  1775. return queryBase()->next();
  1776. }
  1777. bool IndirectResultSetCursor::previous()
  1778. {
  1779. return queryBase()->previous();
  1780. }
  1781. INewResultSet * IndirectResultSetCursor::queryResultSet()
  1782. {
  1783. return queryBase()->queryResultSet();
  1784. }
  1785. bool IndirectResultSetCursor::relative(__int64 rows)
  1786. {
  1787. return queryBase()->relative(rows);
  1788. }
  1789. void IndirectResultSetCursor::serialize(IDataVal & d)
  1790. {
  1791. queryBase()->serialize(d);
  1792. }
  1793. IStringVal & IndirectResultSetCursor::getDisplayText(IStringVal &ret, int columnIndex)
  1794. {
  1795. return queryBase()->getDisplayText(ret, columnIndex);
  1796. }
  1797. IStringVal & IndirectResultSetCursor::getXml(IStringVal & ret, int columnIndex)
  1798. {
  1799. return queryBase()->getXml(ret, columnIndex);
  1800. }
  1801. IStringVal & IndirectResultSetCursor::getXmlRow(IStringVal &ret)
  1802. {
  1803. return queryBase()->getXmlRow(ret);
  1804. }
  1805. IStringVal & IndirectResultSetCursor::getXmlItem(IStringVal &ret)
  1806. {
  1807. return queryBase()->getXmlItem(ret);
  1808. }
  1809. void IndirectResultSetCursor::beginWriteXmlRows(IXmlWriter & writer)
  1810. {
  1811. return queryBase()->beginWriteXmlRows(writer);
  1812. }
  1813. void IndirectResultSetCursor::writeXmlRow(IXmlWriter &writer)
  1814. {
  1815. return queryBase()->writeXmlRow(writer);
  1816. }
  1817. void IndirectResultSetCursor::endWriteXmlRows(IXmlWriter & writer)
  1818. {
  1819. return queryBase()->endWriteXmlRows(writer);
  1820. }
  1821. void IndirectResultSetCursor::writeXmlItem(IXmlWriter &writer)
  1822. {
  1823. return queryBase()->writeXmlItem(writer);
  1824. }
  1825. void IndirectResultSetCursor::noteRelatedFileChanged()
  1826. {
  1827. queryBase()->noteRelatedFileChanged();
  1828. }
  1829. //---------------------------------------------------------------------------
  1830. bool NotifyingResultSetCursor::absolute(__int64 row)
  1831. {
  1832. bool ret = IndirectResultSetCursor::absolute(row);
  1833. notifyChanged();
  1834. return ret;
  1835. }
  1836. void NotifyingResultSetCursor::afterLast()
  1837. {
  1838. IndirectResultSetCursor::afterLast();
  1839. notifyChanged();
  1840. }
  1841. void NotifyingResultSetCursor::beforeFirst()
  1842. {
  1843. IndirectResultSetCursor::beforeFirst();
  1844. notifyChanged();
  1845. }
  1846. bool NotifyingResultSetCursor::fetch(__int64 fileoffset)
  1847. {
  1848. bool ret = IndirectResultSetCursor::fetch(fileoffset);
  1849. notifyChanged();
  1850. return ret;
  1851. }
  1852. bool NotifyingResultSetCursor::first()
  1853. {
  1854. bool ret = IndirectResultSetCursor::first();
  1855. notifyChanged();
  1856. return ret;
  1857. }
  1858. bool NotifyingResultSetCursor::last()
  1859. {
  1860. bool ret = IndirectResultSetCursor::last();
  1861. notifyChanged();
  1862. return ret;
  1863. }
  1864. bool NotifyingResultSetCursor::next()
  1865. {
  1866. bool ret = IndirectResultSetCursor::next();
  1867. notifyChanged();
  1868. return ret;
  1869. }
  1870. bool NotifyingResultSetCursor::previous()
  1871. {
  1872. bool ret = IndirectResultSetCursor::previous();
  1873. notifyChanged();
  1874. return ret;
  1875. }
  1876. bool NotifyingResultSetCursor::relative(__int64 rows)
  1877. {
  1878. bool ret = IndirectResultSetCursor::relative(rows);
  1879. notifyChanged();
  1880. return ret;
  1881. }
  1882. void NotifyingResultSetCursor::noteRelatedFileChanged()
  1883. {
  1884. IndirectResultSetCursor::noteRelatedFileChanged();
  1885. notifyChanged();
  1886. }
  1887. void NotifyingResultSetCursor::notifyChanged()
  1888. {
  1889. ForEachItemIn(i, dependents)
  1890. dependents.item(i).noteRelatedFileChanged();
  1891. }
  1892. //---------------------------------------------------------------------------
  1893. DelayedFilteredResultSetCursor::DelayedFilteredResultSetCursor(INewResultSet * _resultSet)
  1894. {
  1895. filtered.setown(_resultSet->createFiltered());
  1896. }
  1897. void DelayedFilteredResultSetCursor::clearCursor()
  1898. {
  1899. cursor.clear();
  1900. resultSet.clear();
  1901. }
  1902. void DelayedFilteredResultSetCursor::clearFilters()
  1903. {
  1904. clearCursor();
  1905. filtered->clearFilters();
  1906. }
  1907. void DelayedFilteredResultSetCursor::ensureFiltered()
  1908. {
  1909. }
  1910. void DelayedFilteredResultSetCursor::noteRelatedFileChanged()
  1911. {
  1912. //Don't create a cursor, just to tell the class that the cursor is no longer valid!
  1913. if (cursor)
  1914. IndirectResultSetCursor::noteRelatedFileChanged();
  1915. }
  1916. IExtendedResultSetCursor * DelayedFilteredResultSetCursor::queryBase()
  1917. {
  1918. //NB: Not thread safe - but none of the interface is
  1919. if (!cursor)
  1920. {
  1921. ensureFiltered();
  1922. //MORE: should possibly have the ability to create a null dataset at this point.
  1923. resultSet.setown(filtered->create());
  1924. cursor.setown(static_cast<IExtendedResultSetCursor *>(resultSet->createCursor()));
  1925. }
  1926. return cursor;
  1927. }
  1928. //---------------------------------------------------------------------------
  1929. //was using function templates that didn't use their types in their arguments, but
  1930. //VC++ fails to process them correctly.
  1931. template <class KEY> int qsortCompare(const void * _left, const void * _right) { return 0; }
  1932. typedef int (*qsortFunc)(const void *, const void *);
  1933. struct StringKeyElement
  1934. {
  1935. unsigned index;
  1936. StringAttr value;
  1937. };
  1938. int qsortCompare_StringKeyElement(const void * _left, const void * _right)
  1939. {
  1940. const StringKeyElement * left = (const StringKeyElement *)_left;
  1941. const StringKeyElement * right = (const StringKeyElement *)_right;
  1942. int i = strcmp(left->value, right->value);
  1943. if (i != 0)
  1944. return i;
  1945. if (left->index < right->index) return -1;
  1946. assertex(left->index > right->index);
  1947. return +1;
  1948. }
  1949. qsortFunc qsortCompare(StringKeyElement *) { return qsortCompare_StringKeyElement; }
  1950. void extractKey(StringKeyElement & element, CResultSetCursor * cursor, unsigned column)
  1951. {
  1952. StringAttrAdaptor adaptor(element.value);
  1953. cursor->getString(adaptor, column);
  1954. }
  1955. struct IntegerKeyElement
  1956. {
  1957. unsigned index;
  1958. __int64 value;
  1959. };
  1960. int qsortCompare_IntegerKeyElement(const void * _left, const void * _right)
  1961. {
  1962. const IntegerKeyElement * left = (const IntegerKeyElement *)_left;
  1963. const IntegerKeyElement * right = (const IntegerKeyElement *)_right;
  1964. if (left->value < right->value) return -1;
  1965. if (left->value > right->value) return +1;
  1966. if (left->index < right->index) return -1;
  1967. assertex(left->index > right->index);
  1968. return +1;
  1969. }
  1970. qsortFunc qsortCompare(IntegerKeyElement *) { return qsortCompare_IntegerKeyElement; }
  1971. void extractKey(IntegerKeyElement & element, CResultSetCursor * cursor, unsigned column)
  1972. {
  1973. element.value = cursor->getInt(column);
  1974. }
  1975. struct UnsignedKeyElement
  1976. {
  1977. unsigned index;
  1978. unsigned __int64 value;
  1979. };
  1980. int qsortCompare_UnsignedKeyElement(const void * _left, const void * _right)
  1981. {
  1982. const UnsignedKeyElement * left = (const UnsignedKeyElement *)_left;
  1983. const UnsignedKeyElement * right = (const UnsignedKeyElement *)_right;
  1984. if (left->value < right->value) return -1;
  1985. if (left->value > right->value) return +1;
  1986. if (left->index < right->index) return -1;
  1987. assertex(left->index > right->index);
  1988. return +1;
  1989. }
  1990. qsortFunc qsortCompare(UnsignedKeyElement *) { return qsortCompare_UnsignedKeyElement; }
  1991. void extractKey(UnsignedKeyElement & element, CResultSetCursor * cursor, unsigned column)
  1992. {
  1993. element.value = (unsigned __int64)cursor->getInt(column);
  1994. }
  1995. struct RealKeyElement
  1996. {
  1997. unsigned index;
  1998. double value;
  1999. };
  2000. int qsortCompare_RealKeyElement(const void * _left, const void * _right)
  2001. {
  2002. const RealKeyElement * left = (const RealKeyElement *)_left;
  2003. const RealKeyElement * right = (const RealKeyElement *)_right;
  2004. if (left->value < right->value) return -1;
  2005. if (left->value > right->value) return +1;
  2006. if (left->index < right->index) return -1;
  2007. assertex(left->index > right->index);
  2008. return +1;
  2009. }
  2010. qsortFunc qsortCompare(RealKeyElement *) { return qsortCompare_RealKeyElement; }
  2011. void extractKey(RealKeyElement & element, CResultSetCursor * cursor, unsigned column)
  2012. {
  2013. element.value = cursor->getDouble(column);
  2014. }
  2015. template <class KEYELEMENT>
  2016. void buildCursorIndex(unsigned & numElements, unsigned * & elements, CResultSetCursor * cursor, unsigned column, bool descend, KEYELEMENT *)
  2017. {
  2018. __int64 max64 = cursor->getNumRows();
  2019. assertex(max64 <= MAX_SORT_ELEMENTS);
  2020. unsigned max = (unsigned)max64;
  2021. KEYELEMENT * keys = new KEYELEMENT[max];
  2022. for (unsigned idx=0; idx < max; idx++)
  2023. {
  2024. cursor->absolute(idx);
  2025. keys[idx].index = idx;
  2026. extractKey(keys[idx], cursor, column);
  2027. }
  2028. qsort(keys, max, sizeof(KEYELEMENT), qsortCompare((KEYELEMENT *)0));
  2029. numElements = max;
  2030. elements = (unsigned *)malloc(max * sizeof(unsigned));
  2031. if (descend)
  2032. {
  2033. for (unsigned idx=0; idx < max; idx++)
  2034. elements[max-idx-1] = keys[idx].index;
  2035. }
  2036. else
  2037. {
  2038. for (unsigned idx=0; idx < max; idx++)
  2039. elements[idx] = keys[idx].index;
  2040. }
  2041. delete [] keys;
  2042. }
  2043. CResultSetSortedCursor::CResultSetSortedCursor(const CResultSetMetaData & _meta, IExtendedNewResultSet * _resultSet, unsigned _column, bool _desc) : CResultSetCursor(_meta, _resultSet)
  2044. {
  2045. numEntries = 0;
  2046. elements = NULL;
  2047. lastRow = 0;
  2048. column = _column;
  2049. desc = _desc;
  2050. buildIndex();
  2051. }
  2052. CResultSetSortedCursor::CResultSetSortedCursor(const CResultSetMetaData & _meta, IExtendedNewResultSet * _resultSet, unsigned _column, bool _desc, MemoryBuffer & buffer) : CResultSetCursor(_meta, _resultSet)
  2053. {
  2054. numEntries = 0;
  2055. elements = NULL;
  2056. lastRow = 0;
  2057. column = _column;
  2058. desc = _desc;
  2059. buildIndex();
  2060. buffer.read(curRow);
  2061. buffer.read(lastRow);
  2062. absolute(lastRow);
  2063. }
  2064. CResultSetSortedCursor::~CResultSetSortedCursor()
  2065. {
  2066. free(elements);
  2067. }
  2068. void CResultSetSortedCursor::buildIndex()
  2069. {
  2070. Owned<CResultSetCursor> cursor = new CResultSetCursor(meta, resultSet);
  2071. switch (meta.getColumnDisplayType(column))
  2072. {
  2073. case TypeBoolean:
  2074. case TypeInteger:
  2075. buildCursorIndex<IntegerKeyElement>(numEntries, elements, cursor, column, desc, (IntegerKeyElement*)0);
  2076. break;
  2077. case TypeUnsignedInteger:
  2078. buildCursorIndex<UnsignedKeyElement>(numEntries, elements, cursor, column, desc, (UnsignedKeyElement*)0);
  2079. break;
  2080. case TypeReal:
  2081. buildCursorIndex<RealKeyElement>(numEntries, elements, cursor, column, desc, (RealKeyElement*)0);
  2082. break;
  2083. case TypeString:
  2084. case TypeData:
  2085. case TypeUnicode: //MORE!
  2086. buildCursorIndex<StringKeyElement>(numEntries, elements, cursor, column, desc, (StringKeyElement*)0);
  2087. break;
  2088. default:
  2089. UNIMPLEMENTED; // Should have been translated to one of the above by this point...
  2090. }
  2091. }
  2092. bool CResultSetSortedCursor::absolute(__int64 row)
  2093. {
  2094. lastRow = row;
  2095. return CResultSetCursor::absolute(translateRow(row));
  2096. }
  2097. __int64 CResultSetSortedCursor::getCurRow() const
  2098. {
  2099. return lastRow;
  2100. }
  2101. __int64 CResultSetSortedCursor::translateRow(__int64 row) const
  2102. {
  2103. if ((row >= 0) && (row < numEntries))
  2104. return elements[row];
  2105. return row;
  2106. }
  2107. void CResultSetSortedCursor::serializeType(MemoryBuffer & buffer)
  2108. {
  2109. buffer.append((unsigned)column);
  2110. buffer.append(desc);
  2111. }
  2112. void CResultSetSortedCursor::serialize(MemoryBuffer & buffer)
  2113. {
  2114. CResultSetCursor::serialize(buffer);
  2115. buffer.append(lastRow);
  2116. }
  2117. //---------------------------------------------------------------------------
  2118. inline byte hex2digit(char c)
  2119. {
  2120. if (c >= 'a')
  2121. return (c - 'a' + 10);
  2122. else if (c >= 'A')
  2123. return (c - 'A' + 10);
  2124. return (c - '0');
  2125. }
  2126. inline byte getHexPair(const char * s)
  2127. {
  2128. return hex2digit(s[0]) << 4 | hex2digit(s[1]);
  2129. }
  2130. static unsigned getSubstringMatchLength(size32_t len, const void * data)
  2131. {
  2132. const char * inbuff = (const char *)data;
  2133. unsigned trimLen = rtlTrimStrLen(len, inbuff);
  2134. if (trimLen && (inbuff[trimLen-1] == '*'))
  2135. return rtlUtf8Length(trimLen-1, inbuff);
  2136. return FullStringMatch;
  2137. }
  2138. void CColumnFilter::addValue(unsigned sizeText, const char * text)
  2139. {
  2140. unsigned lenText = rtlUtf8Length(sizeText, text);
  2141. unsigned size = type->getSize();
  2142. MemoryAttrItem * next = new MemoryAttrItem;
  2143. type_t tc = type->getTypeCode();
  2144. if (isMappedIndexField)
  2145. {
  2146. if (__BYTE_ORDER == __LITTLE_ENDIAN)
  2147. {
  2148. if (tc == type_int)
  2149. tc = type_swapint;
  2150. }
  2151. else
  2152. {
  2153. if (tc == type_swapint)
  2154. tc = type_int;
  2155. }
  2156. }
  2157. //sublen is the number of characters (not the size)
  2158. subLen = getSubstringMatchLength(sizeText, text);
  2159. switch (tc)
  2160. {
  2161. case type_void:
  2162. case type_set:
  2163. case type_table:
  2164. case type_groupedtable:
  2165. break;
  2166. case type_boolean:
  2167. {
  2168. byte value = rtlCsvStrToBool(lenText, text);
  2169. next->set(sizeof(value), &value);
  2170. break;
  2171. }
  2172. break;
  2173. case type_int:
  2174. {
  2175. __int64 value = type->isSigned() ? rtlStrToInt8(lenText, text) : (__int64)rtlStrToUInt8(lenText, text);
  2176. if (isMappedIndexField && type->isSigned())
  2177. value += getIntBias(size);
  2178. if (__BYTE_ORDER == __LITTLE_ENDIAN)
  2179. next->set(size, &value);
  2180. else
  2181. next->set(size, ((const byte *)&value)+(sizeof(value)-size));
  2182. break;
  2183. }
  2184. case type_swapint:
  2185. {
  2186. __int64 value = type->isSigned() ? rtlStrToInt8(lenText, text) : (__int64)rtlStrToUInt8(lenText, text);
  2187. if (isMappedIndexField && type->isSigned())
  2188. value += getIntBias(size);
  2189. _rev8((char *)&value);
  2190. if (__BYTE_ORDER == __LITTLE_ENDIAN)
  2191. next->set(size, ((const byte *)&value)+(sizeof(value)-size));
  2192. else
  2193. next->set(size, &value);
  2194. break;
  2195. }
  2196. case type_packedint:
  2197. {
  2198. void * target = next->allocate(size);
  2199. if (type->isSigned())
  2200. rtlSetPackedSigned(target, rtlStrToInt8(lenText, text));
  2201. else
  2202. rtlSetPackedUnsigned(target, rtlStrToUInt8(lenText, text));
  2203. break;
  2204. }
  2205. case type_decimal:
  2206. {
  2207. void * target = next->allocate(size);
  2208. DecLock();
  2209. rtlDecPushUtf8(lenText, text);
  2210. if (type->isSigned())
  2211. DecPopDecimal(target, size, type->getPrecision());
  2212. else
  2213. DecPopUDecimal(target, size, type->getPrecision());
  2214. DecUnlock();
  2215. break;
  2216. }
  2217. case type_real:
  2218. {
  2219. if (size == 4)
  2220. {
  2221. float value = (float)rtlStrToReal(lenText, text);
  2222. next->set(sizeof(value), &value);
  2223. }
  2224. else
  2225. {
  2226. double value = rtlStrToReal(lenText, text);
  2227. next->set(sizeof(value), &value);
  2228. }
  2229. break;
  2230. }
  2231. case type_qstring:
  2232. {
  2233. if (lenText > subLen) lenText = subLen;
  2234. char * target = (char *)next->allocate(rtlQStrSize(lenText));
  2235. rtlStrToQStr(lenText, target, lenText, text);
  2236. break;
  2237. }
  2238. break;
  2239. case type_data:
  2240. {
  2241. if (lenText > subLen) lenText = subLen;
  2242. if (subLen != FullStringMatch)
  2243. subLen /= 2;
  2244. unsigned max = lenText/2;
  2245. char * target = (char *)next->allocate(max);
  2246. for (unsigned i=0; i<max; i++)
  2247. target[i] = getHexPair(text+i*2);
  2248. break;
  2249. }
  2250. case type_string:
  2251. {
  2252. if (lenText > subLen) lenText = subLen;
  2253. char * target = (char *)next->allocate(lenText);
  2254. rtlUtf8ToStr(lenText, target, lenText, text);
  2255. break;
  2256. }
  2257. case type_unicode:
  2258. {
  2259. if (lenText > subLen) lenText = subLen;
  2260. UChar * target = (UChar *)next->allocate(lenText*2);
  2261. rtlUtf8ToUnicode(lenText, target, lenText, text);
  2262. break;
  2263. }
  2264. break;
  2265. case type_varstring:
  2266. {
  2267. if (lenText > subLen) lenText = subLen;
  2268. next->set(lenText+1, text);
  2269. break;
  2270. }
  2271. case type_varunicode:
  2272. {
  2273. UChar * target = (UChar *)next->allocate(lenText*2+2);
  2274. rtlUtf8ToUnicode(lenText, target, lenText, text);
  2275. target[lenText] = 0;
  2276. break;
  2277. }
  2278. break;
  2279. case type_utf8:
  2280. {
  2281. if (lenText > subLen) sizeText = rtlUtf8Size(subLen, text);
  2282. next->set(sizeText, text);
  2283. //Should it be utf8 or ascii coming in?
  2284. //char * target = (char *)next->allocate(lenText*4);
  2285. //rtlStrToUtf8(lenText, target, lenText, text);
  2286. break;
  2287. }
  2288. break;
  2289. default:
  2290. UNIMPLEMENTED;
  2291. }
  2292. if (next->length())
  2293. values.append(*next);
  2294. else
  2295. next->Release();
  2296. }
  2297. void CColumnFilter::deserialize(MemoryBuffer & buffer)
  2298. {
  2299. unsigned num;
  2300. buffer.read(num);
  2301. for (unsigned i= 0; i < num; i++)
  2302. {
  2303. MemoryAttrItem * next = new MemoryAttrItem;
  2304. ::deserialize(buffer, *next);
  2305. values.append(*next);
  2306. }
  2307. }
  2308. bool CColumnFilter::optimizeFilter(IFvDataSource * dataSource)
  2309. {
  2310. if (values.ordinality() == 0)
  2311. return true;
  2312. optimized = true;
  2313. ForEachItemIn(i, values)
  2314. {
  2315. MemoryAttr & cur = values.item(i);
  2316. if (!dataSource->addFilter(whichColumn, subLen, cur.length(), cur.get()))
  2317. optimized = false;
  2318. }
  2319. return optimized;
  2320. }
  2321. bool CColumnFilter::matches(const byte * rowValue, unsigned valueSize, const byte * value)
  2322. {
  2323. unsigned size = type->getSize();
  2324. unsigned len;
  2325. switch (type->getTypeCode())
  2326. {
  2327. case type_void:
  2328. case type_set:
  2329. case type_table:
  2330. case type_groupedtable:
  2331. return true;
  2332. case type_boolean:
  2333. case type_int:
  2334. case type_swapint:
  2335. case type_packedint:
  2336. return memcmp(rowValue, value, size) == 0;
  2337. case type_decimal:
  2338. if (type->isSigned())
  2339. return DecCompareDecimal(size, rowValue, value) == 0;
  2340. return DecCompareUDecimal(size, rowValue, value) == 0;
  2341. case type_real:
  2342. if (size == 4)
  2343. return *(float *)rowValue == *(float *)value;
  2344. return *(double *)rowValue == *(double *)value;
  2345. case type_qstring:
  2346. len = getLength(*type, rowValue);
  2347. if ((subLen != FullStringMatch) && (len > subLen))
  2348. len = subLen;
  2349. return rtlCompareQStrQStr(len, rowValue, rtlQStrLength(valueSize), value) == 0;
  2350. case type_data:
  2351. len = getLength(*type, rowValue);
  2352. if ((subLen != FullStringMatch) && (len > subLen))
  2353. len = subLen;
  2354. return rtlCompareDataData(len, (const char *)rowValue, valueSize, (const char *)value) == 0;
  2355. case type_string:
  2356. len = getLength(*type, rowValue);
  2357. if ((subLen != FullStringMatch) && (len > subLen))
  2358. len = subLen;
  2359. return rtlCompareStrStr(len, (const char *)rowValue, valueSize, (const char *)value) == 0;
  2360. case type_unicode:
  2361. len = getLength(*type, rowValue);
  2362. return rtlCompareUnicodeUnicode(len, (const UChar *)rowValue, valueSize/2, (const UChar *)value, "") == 0;
  2363. case type_utf8:
  2364. len = getLength(*type, rowValue);
  2365. if ((subLen != FullStringMatch) && (len > subLen))
  2366. len = subLen;
  2367. return rtlCompareUtf8Utf8(len, (const char *)rowValue, rtlUtf8Length(valueSize, value), (const char *)value, "") == 0;
  2368. case type_varstring:
  2369. return strcmp((const char *)rowValue, (const char *)value) != 0;
  2370. case type_varunicode:
  2371. return rtlCompareVUnicodeVUnicode((const UChar *)rowValue, (const UChar *)value, "") == 0;
  2372. default:
  2373. UNIMPLEMENTED;
  2374. }
  2375. }
  2376. bool CColumnFilter::isValid(const byte * rowValue, const unsigned * offsets)
  2377. {
  2378. if (optimized || values.ordinality() == 0)
  2379. return true;
  2380. const byte * columnValue = rowValue + offsets[whichColumn];
  2381. ForEachItemIn(i, values)
  2382. {
  2383. MemoryAttr & cur = values.item(i);
  2384. if (matches(columnValue, cur.length(), (const byte *)cur.get()))
  2385. return true;
  2386. }
  2387. return false;
  2388. }
  2389. void CColumnFilter::serialize(MemoryBuffer & buffer)
  2390. {
  2391. buffer.append((unsigned)values.ordinality());
  2392. ForEachItemIn(i, values)
  2393. ::serialize(buffer, values.item(i));
  2394. }
  2395. //---------------------------------------------------------------------------
  2396. CFilteredResultSet::CFilteredResultSet(IExtendedNewResultSet * _other, ColumnFilterArray & _filters) : CIndirectResultSet(_other)
  2397. {
  2398. appendArray(filters, _filters);
  2399. initExtra();
  2400. }
  2401. #if 0
  2402. CFilteredResultSet::CFilteredResultSet(const CResultSetMetaData & _meta, CResultSet * _resultSet, MemoryBuffer & buffer) : CResultSetCursor(_meta, _resultSet, false)
  2403. {
  2404. unsigned numFilters;
  2405. buffer.read(numFilters);
  2406. for (unsigned i=0; i< numFilters; i++)
  2407. {
  2408. CColumnFilter & next = * new CColumnFilter(meta.queryType(i));
  2409. next.deserialize(buffer);
  2410. filters.append(next);
  2411. }
  2412. initExtra();
  2413. buffer.read(curRow);
  2414. buffer.read(lastRow);
  2415. absolute(lastRow);
  2416. }
  2417. #endif
  2418. CFilteredResultSet::~CFilteredResultSet()
  2419. {
  2420. delete [] offsets;
  2421. }
  2422. bool CFilteredResultSet::getRow(MemoryBuffer & out, __int64 row)
  2423. {
  2424. __int64 newRow = translateRow(row);
  2425. if (newRow < 0)
  2426. return false;
  2427. return CIndirectResultSet::getRow(out, newRow);
  2428. }
  2429. bool CFilteredResultSet::getRawRow(MemoryBuffer & out, __int64 row)
  2430. {
  2431. __int64 newRow = translateRow(row);
  2432. if (newRow < 0)
  2433. return false;
  2434. return CIndirectResultSet::getRawRow(out, newRow);
  2435. }
  2436. __int64 CFilteredResultSet::getNumRows() const
  2437. {
  2438. if (readAll)
  2439. return validPositions.ordinality();
  2440. return UNKNOWN_NUM_ROWS;
  2441. }
  2442. bool CFilteredResultSet::rowMatchesFilter(const byte * row)
  2443. {
  2444. if (!meta.isFixedSize())
  2445. meta.calcFieldOffsets(row, offsets);
  2446. ForEachItemIn(i, filters)
  2447. {
  2448. if (!filters.item(i).isValid(row, offsets))
  2449. return false;
  2450. }
  2451. return true;
  2452. }
  2453. __int64 CFilteredResultSet::translateRow(__int64 row)
  2454. {
  2455. if (row < 0)
  2456. return row;
  2457. if (row > MAX_FILTER_ELEMENTS)
  2458. return AFTER_LAST_ROW; // would it be better to throw an error?
  2459. if (!readAll && (row >= validPositions.ordinality()))
  2460. {
  2461. unsigned __int64 nextPos = 0;
  2462. if (validPositions.ordinality())
  2463. nextPos = validPositions.tos()+1;
  2464. MemoryBuffer tempBuffer;
  2465. unsigned startTime = msTick();
  2466. while (row >= validPositions.ordinality())
  2467. {
  2468. if (!CIndirectResultSet::getRow(tempBuffer.clear(), nextPos))
  2469. {
  2470. readAll = true;
  2471. break;
  2472. }
  2473. if (rowMatchesFilter((byte *)tempBuffer.toByteArray()))
  2474. {
  2475. validPositions.append(nextPos);
  2476. }
  2477. else
  2478. {
  2479. unsigned timeTaken = msTick() - startTime;
  2480. if (timeTaken > MAX_SKIP_TIME)
  2481. throwError1(FVERR_FilterTooRestrictive, timeTaken/1000);
  2482. }
  2483. nextPos++;
  2484. }
  2485. }
  2486. if ((unsigned)row < validPositions.ordinality())
  2487. return validPositions.item((unsigned)row);
  2488. return AFTER_LAST_ROW;
  2489. }
  2490. void CFilteredResultSet::serialize(MemoryBuffer & buffer)
  2491. {
  2492. CIndirectResultSet::serialize(buffer);
  2493. buffer.append((unsigned)filters.ordinality());
  2494. ForEachItemIn(i, filters)
  2495. filters.item(i).serialize(buffer);
  2496. }
  2497. void CFilteredResultSet::initExtra()
  2498. {
  2499. readAll = false;
  2500. offsets = new unsigned[meta.getColumnCount()+1];
  2501. if (meta.isFixedSize())
  2502. meta.calcFieldOffsets(NULL, offsets);
  2503. }
  2504. //---------------------------------------------------------------------------
  2505. CFetchFilteredResultSet::CFetchFilteredResultSet(IExtendedNewResultSet * _parent, const CStrColumnFilter & _filter) : CIndirectResultSet(_parent)
  2506. {
  2507. ForEachItemIn(i, _filter.values)
  2508. {
  2509. const MemoryAttrItem & value = _filter.values.item(i);
  2510. unsigned __int64 offset = rtlStrToUInt8(value.length(), static_cast<const char *>(value.get()));
  2511. validOffsets.append(offset);
  2512. }
  2513. }
  2514. bool CFetchFilteredResultSet::getRow(MemoryBuffer & out, __int64 row)
  2515. {
  2516. if (!validOffsets.isItem((unsigned)row))
  2517. return false;
  2518. return CIndirectResultSet::fetch(out, validOffsets.item((unsigned)row));
  2519. }
  2520. bool CFetchFilteredResultSet::getRawRow(MemoryBuffer & out, __int64 row)
  2521. {
  2522. if (!validOffsets.isItem((unsigned)row))
  2523. return false;
  2524. return CIndirectResultSet::fetchRaw(out, validOffsets.item((unsigned)row));
  2525. }
  2526. __int64 CFetchFilteredResultSet::getNumRows() const
  2527. {
  2528. return validOffsets.ordinality();
  2529. }
  2530. void CFetchFilteredResultSet::serialize(MemoryBuffer & buffer)
  2531. {
  2532. CIndirectResultSet::serialize(buffer);
  2533. buffer.append((unsigned)validOffsets.ordinality());
  2534. ForEachItemIn(i, validOffsets)
  2535. buffer.append(validOffsets.item(i));
  2536. }
  2537. //---------------------------------------------------------------------------
  2538. CFilteredResultSetBuilder::CFilteredResultSetBuilder(IExtendedNewResultSet * _resultSet)
  2539. {
  2540. resultSet.set(_resultSet);
  2541. }
  2542. INewResultSet * CFilteredResultSetBuilder::create()
  2543. {
  2544. Linked<IExtendedNewResultSet> baseResultSet = resultSet;
  2545. const CResultSetMetaData & meta = static_cast<const CResultSetMetaData &>(resultSet->getMetaData());
  2546. //Check for fetch filter, and if present apply that first
  2547. unsigned numColumns = meta.getColumnCount();
  2548. if (filters.isItem(numColumns-1) && meta.isVirtual(numColumns-1))
  2549. {
  2550. CStrColumnFilter & cur = filters.item(numColumns-1);
  2551. if (cur.values.ordinality())
  2552. {
  2553. baseResultSet.setown(new CFetchFilteredResultSet(resultSet, cur));
  2554. cur.values.kill();
  2555. }
  2556. }
  2557. Owned<IExtendedNewResultSet> cloned = baseResultSet->cloneForFilter();
  2558. IFvDataSource * dataSource = cloned ? cloned->queryDataSource() : NULL;
  2559. ColumnFilterArray rawFilters;
  2560. ForEachItemIn(columnIndex, filters)
  2561. {
  2562. CStrColumnFilter & cur = filters.item(columnIndex);
  2563. if (cur.values.ordinality())
  2564. {
  2565. Owned<CColumnFilter> next = new CColumnFilter(columnIndex, meta.queryType(columnIndex), resultSet->isMappedIndexField(columnIndex));
  2566. ForEachItemIn(j, cur.values)
  2567. {
  2568. MemoryAttrItem & curValue = cur.values.item(j);
  2569. next->addValue(curValue.length(), reinterpret_cast<const char *>(curValue.get()));
  2570. }
  2571. if (!cloned || !next->optimizeFilter(dataSource))
  2572. rawFilters.append(*next.getClear());
  2573. }
  2574. }
  2575. if (cloned)
  2576. {
  2577. dataSource->applyFilter();
  2578. if (rawFilters.ordinality() == 0)
  2579. return cloned.getClear();
  2580. return new CFilteredResultSet(cloned, rawFilters);
  2581. }
  2582. if (rawFilters.ordinality() == 0)
  2583. return baseResultSet.getClear();
  2584. return new CFilteredResultSet(baseResultSet, rawFilters);
  2585. }
  2586. void CFilteredResultSetBuilder::addFilter(unsigned columnIndex, const char * value)
  2587. {
  2588. addFilter(columnIndex, strlen(value), value);
  2589. }
  2590. void CFilteredResultSetBuilder::addFilter(unsigned columnIndex, unsigned len, const char * value)
  2591. {
  2592. assertex(columnIndex < (unsigned)resultSet->getMetaData().getColumnCount());
  2593. while (filters.ordinality() <= columnIndex)
  2594. filters.append(*new CStrColumnFilter());
  2595. filters.item(columnIndex).addValue(len, value);
  2596. }
  2597. void CFilteredResultSetBuilder::addNaturalFilter(unsigned columnIndex, unsigned len, const char * value)
  2598. {
  2599. const CResultSetMetaData & meta = static_cast<const CResultSetMetaData &>(resultSet->getMetaData());
  2600. assertex(columnIndex < (unsigned)meta.getColumnCount());
  2601. const CResultSetColumnInfo & column = meta.queryColumn(columnIndex);
  2602. if (column.setTransforms.ordinality())
  2603. {
  2604. MemoryAttr source(len, value);
  2605. MemoryAttr translated;
  2606. translateValue(translated, source, column.setTransforms);
  2607. addFilter(columnIndex, translated.length(), static_cast<const char *>(translated.get()));
  2608. }
  2609. else
  2610. addFilter(columnIndex, len, value);
  2611. }
  2612. void CFilteredResultSetBuilder::clearFilter(unsigned columnIndex)
  2613. {
  2614. assertex(columnIndex < (unsigned)resultSet->getMetaData().getColumnCount());
  2615. if (filters.isItem(columnIndex))
  2616. filters.item(columnIndex).clear();
  2617. }
  2618. void CFilteredResultSetBuilder::clearFilters()
  2619. {
  2620. filters.kill();
  2621. }
  2622. //---------------------------------------------------------------------------
  2623. CResultSetFactoryBase::CResultSetFactoryBase(const char * _username, const char * _password)
  2624. {
  2625. username.set(_username);
  2626. password.set(_password);
  2627. }
  2628. CResultSetFactoryBase::CResultSetFactoryBase(ISecManager &secmgr, ISecUser &secuser)
  2629. {
  2630. secMgr.set(&secmgr);
  2631. secUser.set(&secuser);
  2632. username.set(secuser.getName());
  2633. password.set(secuser.credentials().getPassword());
  2634. }
  2635. //---------------------------------------------------------------------------
  2636. CResultSetFactory::CResultSetFactory(const char * _username, const char * _password) : CResultSetFactoryBase(_username, _password)
  2637. {
  2638. }
  2639. CResultSetFactory::CResultSetFactory(ISecManager &secmgr, ISecUser &secuser) : CResultSetFactoryBase(secmgr, secuser)
  2640. {
  2641. }
  2642. IDistributedFile * CResultSetFactory::lookupLogicalName(const char * logicalName)
  2643. {
  2644. Owned<IUserDescriptor> udesc;
  2645. if(username != NULL && *username != '\0')
  2646. {
  2647. udesc.setown(createUserDescriptor());
  2648. udesc->set(username, password);
  2649. }
  2650. Owned<IDistributedFile> df = queryDistributedFileDirectory().lookup(logicalName, udesc.get());
  2651. if (!df)
  2652. throwError1(FVERR_CouldNotResolveX, logicalName);
  2653. return df.getClear();
  2654. }
  2655. INewResultSet * CResultSetFactory::createNewResultSet(IConstWUResult * wuResult, const char * wuid)
  2656. {
  2657. Owned<IFvDataSource> ds = createDataSource(wuResult, wuid, username, password);
  2658. if (ds)
  2659. return createResultSet(ds, (wuResult->getResultFormat() == ResultFormatXml));
  2660. return NULL;
  2661. }
  2662. INewResultSet * CResultSetFactory::createNewFileResultSet(const char * logicalName, const char * cluster)
  2663. {
  2664. Owned<IDistributedFile> df = lookupLogicalName(logicalName);
  2665. Owned<IFvDataSource> ds = createFileDataSource(df, logicalName, cluster, username, password);
  2666. if (ds)
  2667. {
  2668. Owned<CResultSet> result = createResultSet(ds, false);
  2669. result->setColumnMapping(df);
  2670. return result.getClear();
  2671. }
  2672. return NULL;
  2673. }
  2674. INewResultSet * CResultSetFactory::createNewResultSet(const char * wuid, unsigned sequence, const char * name)
  2675. {
  2676. Owned<IConstWUResult> wuResult = (secMgr) ? secResolveResult(*secMgr, *secUser, wuid, sequence, name) : resolveResult(wuid, sequence, name);
  2677. return (wuResult) ? createNewResultSet(wuResult, wuid) : NULL;
  2678. }
  2679. INewResultSet * CResultSetFactory::createNewFileResultSet(const char * logicalFile)
  2680. {
  2681. return createNewFileResultSet(logicalFile, NULL);
  2682. }
  2683. CResultSet * CResultSetFactory::createResultSet(IFvDataSource * ds, bool _useXPath)
  2684. {
  2685. //MORE: Save in a hash table, which times out after a certain period...
  2686. return new CResultSet(ds, _useXPath);
  2687. }
  2688. IResultSetMetaData * CResultSetFactory::createResultSetMeta(IConstWUResult * wuResult)
  2689. {
  2690. return new CResultSetMetaData(createMetaData(wuResult), false);
  2691. }
  2692. IResultSetMetaData * CResultSetFactory::createResultSetMeta(const char * wuid, unsigned sequence, const char * name)
  2693. {
  2694. Owned<IConstWUResult> wuResult = (secMgr) ? secResolveResult(*secMgr, *secUser, wuid, sequence, name) : resolveResult(wuid, sequence, name);
  2695. return (wuResult) ? createResultSetMeta(wuResult) : NULL;
  2696. }
  2697. //---------------------------------------------------------------------------
  2698. CRemoteResultSetFactory::CRemoteResultSetFactory(const char * remoteServer, const char * _username, const char * _password) : CResultSetFactoryBase(_username, _password)
  2699. {
  2700. serverEP.set(remoteServer);
  2701. }
  2702. CRemoteResultSetFactory::CRemoteResultSetFactory(const char * remoteServer, ISecManager &secmgr, ISecUser &secuser) : CResultSetFactoryBase(secmgr, secuser)
  2703. {
  2704. serverEP.set(remoteServer);
  2705. }
  2706. INewResultSet * CRemoteResultSetFactory::createNewResultSet(IConstWUResult * wuResult, const char * wuid)
  2707. {
  2708. SCMStringBuffer name;
  2709. wuResult->getResultName(name);
  2710. Owned<IFvDataSource> ds = createRemoteDataSource(serverEP, username, password, wuid, wuResult->getResultSequence(), name.length() ? name.str() : NULL);
  2711. if (ds)
  2712. return new CResultSet(ds, false);
  2713. return NULL;
  2714. }
  2715. INewResultSet * CRemoteResultSetFactory::createNewFileResultSet(const char * logicalName, const char * cluster)
  2716. {
  2717. Owned<IFvDataSource> ds = createRemoteFileDataSource(serverEP, username, password, logicalName);
  2718. if (ds)
  2719. return new CResultSet(ds, false);
  2720. return NULL;
  2721. }
  2722. INewResultSet* CRemoteResultSetFactory::createNewResultSet(const char * wuid, unsigned sequence, const char * name)
  2723. {
  2724. Owned<IFvDataSource> ds = createRemoteDataSource(serverEP, username, password, wuid, sequence, name);
  2725. if (ds)
  2726. return new CResultSet(ds, false);
  2727. return NULL;
  2728. }
  2729. INewResultSet* CRemoteResultSetFactory::createNewFileResultSet(const char * logicalName)
  2730. {
  2731. Owned<IFvDataSource> ds = createRemoteFileDataSource(serverEP, username, password, logicalName);
  2732. if (ds)
  2733. return new CResultSet(ds, false);
  2734. return NULL;
  2735. }
  2736. IResultSetMetaData * CRemoteResultSetFactory::createResultSetMeta(IConstWUResult * wuResult)
  2737. {
  2738. UNIMPLEMENTED;
  2739. }
  2740. IResultSetMetaData * CRemoteResultSetFactory::createResultSetMeta(const char * wuid, unsigned sequence, const char * name)
  2741. {
  2742. UNIMPLEMENTED;
  2743. }
  2744. //---------------------------------------------------------------------------
  2745. INewResultSet* createNewResultSet(IResultSetFactory & factory, IStringVal & error, IConstWUResult * wuResult, const char * wuid)
  2746. {
  2747. try
  2748. {
  2749. return factory.createNewResultSet(wuResult, wuid);
  2750. }
  2751. catch (IException * e)
  2752. {
  2753. StringBuffer s;
  2754. error.set(e->errorMessage(s).str());
  2755. e->Release();
  2756. return NULL;
  2757. }
  2758. }
  2759. INewResultSet* createNewFileResultSet(IResultSetFactory & factory, IStringVal & error, const char * logicalFile, const char * cluster)
  2760. {
  2761. try
  2762. {
  2763. return factory.createNewFileResultSet(logicalFile, cluster);
  2764. }
  2765. catch (IException * e)
  2766. {
  2767. StringBuffer s;
  2768. error.set(e->errorMessage(s).str());
  2769. e->Release();
  2770. return NULL;
  2771. }
  2772. }
  2773. INewResultSet* createNewResultSetSeqName(IResultSetFactory & factory, IStringVal & error, const char * wuid, unsigned sequence, const char * name)
  2774. {
  2775. try
  2776. {
  2777. return factory.createNewResultSet(wuid, sequence, name);
  2778. }
  2779. catch (IException * e)
  2780. {
  2781. StringBuffer s;
  2782. error.set(e->errorMessage(s).str());
  2783. e->Release();
  2784. return NULL;
  2785. }
  2786. }
  2787. //---------------------------------------------------------------------------
  2788. //MORE: There should be an option to create a proxy-based IFileView so that
  2789. //i) formatting etc. can be done remotely
  2790. //ii) cacching and sort order calculation can be done remotely.
  2791. // Should still cache the last n rows locally (raw/non-raw).
  2792. IResultSetFactory * getResultSetFactory(const char * username, const char * password)
  2793. {
  2794. return new CResultSetFactory(username, password);
  2795. }
  2796. IResultSetFactory * getSecResultSetFactory(ISecManager *secmgr, ISecUser *secuser, const char *username, const char *password)
  2797. {
  2798. if (secmgr)
  2799. return new CResultSetFactory(*secmgr, *secuser);
  2800. return getResultSetFactory(username, password);
  2801. }
  2802. IResultSetFactory * getRemoteResultSetFactory(const char * remoteServer, const char * username, const char * password)
  2803. {
  2804. return new CRemoteResultSetFactory(remoteServer, username, password);
  2805. }
  2806. int findResultSetColumn(const INewResultSet * results, const char * columnName)
  2807. {
  2808. const IResultSetMetaData & meta = results->getMetaData();
  2809. SCMStringBuffer s;
  2810. for(int i = 0; i < meta.getColumnCount(); i++)
  2811. {
  2812. s.clear();
  2813. if(!stricmp(columnName, meta.getColumnLabel(s, i).str()))
  2814. return i;
  2815. }
  2816. return -1;
  2817. }
  2818. extern FILEVIEW_API unsigned getResultCursorXml(IStringVal & ret, IResultSetCursor * cursor, const char * name, unsigned start, unsigned count, const char * schemaName, const IProperties *xmlns)
  2819. {
  2820. Owned<CommonXmlWriter> writer = CreateCommonXmlWriter(XWFexpandempty);
  2821. unsigned rc = writeResultCursorXml(*writer, cursor, name, start, count, schemaName, xmlns);
  2822. ret.set(writer->str());
  2823. return rc;
  2824. }
  2825. extern FILEVIEW_API unsigned getResultXml(IStringVal & ret, INewResultSet * result, const char* name,unsigned start, unsigned count, const char * schemaName, const IProperties *xmlns)
  2826. {
  2827. Owned<IResultSetCursor> cursor = result->createCursor();
  2828. return getResultCursorXml(ret, cursor, name, start, count, schemaName, xmlns);
  2829. }
  2830. extern FILEVIEW_API unsigned getResultJSON(IStringVal & ret, INewResultSet * result, const char* name,unsigned start, unsigned count, const char * schemaName)
  2831. {
  2832. Owned<IResultSetCursor> cursor = result->createCursor();
  2833. Owned<CommonJsonWriter> writer = new CommonJsonWriter(0);
  2834. writer->outputBeginRoot();
  2835. unsigned rc = writeResultCursorXml(*writer, cursor, name, start, count, schemaName);
  2836. writer->outputEndRoot();
  2837. ret.set(writer->str());
  2838. return rc;
  2839. }
  2840. extern FILEVIEW_API unsigned writeResultCursorXml(IXmlWriter & writer, IResultSetCursor * cursor, const char * name, unsigned start, unsigned count, const char * schemaName, const IProperties *xmlns)
  2841. {
  2842. if (schemaName)
  2843. {
  2844. writer.outputBeginNested("XmlSchema", false);
  2845. writer.outputUtf8(strlen(schemaName), schemaName, "@name");
  2846. SCMStringBuffer xsd;
  2847. const IResultSetMetaData & meta = cursor->queryResultSet()->getMetaData();
  2848. meta.getXmlXPathSchema(xsd, false);
  2849. writer.outputInlineXml(xsd.str());
  2850. writer.outputEndNested("XmlSchema");
  2851. }
  2852. writer.outputBeginDataset(name, true);
  2853. if (schemaName)
  2854. writer.outputCString(schemaName, "@xmlSchema");
  2855. if (xmlns)
  2856. {
  2857. Owned<IPropertyIterator> it = const_cast<IProperties*>(xmlns)->getIterator();
  2858. ForEach(*it)
  2859. {
  2860. const char *name = it->getPropKey();
  2861. writer.outputXmlns(name,const_cast<IProperties*>(xmlns)->queryProp(name));
  2862. }
  2863. }
  2864. cursor->beginWriteXmlRows(writer);
  2865. unsigned c=0;
  2866. for(bool ok=cursor->absolute(start);ok;ok=cursor->next())
  2867. {
  2868. cursor->writeXmlRow(writer);
  2869. c++;
  2870. if(count && c>=count)
  2871. break;
  2872. }
  2873. cursor->endWriteXmlRows(writer);
  2874. writer.outputEndDataset(name);
  2875. return c;
  2876. }
  2877. extern FILEVIEW_API unsigned writeResultXml(IXmlWriter & writer, INewResultSet * result, const char* name,unsigned start, unsigned count, const char * schemaName, const IProperties *xmlns)
  2878. {
  2879. Owned<IResultSetCursor> cursor = result->createCursor();
  2880. return writeResultCursorXml(writer, cursor, name, start, count, schemaName, xmlns);
  2881. }
  2882. extern FILEVIEW_API unsigned getResultCursorBin(MemoryBuffer & ret, IResultSetCursor * cursor, unsigned start, unsigned count)
  2883. {
  2884. const IResultSetMetaData & meta = cursor->queryResultSet()->getMetaData();
  2885. unsigned numCols = meta.getColumnCount();
  2886. unsigned c=0;
  2887. for(bool ok=cursor->absolute(start);ok;ok=cursor->next())
  2888. {
  2889. for (unsigned col=0; col < numCols; col++)
  2890. cursor->getRaw(MemoryBuffer2IDataVal(ret), col);
  2891. c++;
  2892. if(count && c>=count)
  2893. break;
  2894. }
  2895. return c;
  2896. }
  2897. extern FILEVIEW_API unsigned getResultBin(MemoryBuffer & ret, INewResultSet * result, unsigned start, unsigned count)
  2898. {
  2899. Owned<IResultSetCursor> cursor = result->createCursor();
  2900. return getResultCursorBin(ret, cursor, start, count);
  2901. }
  2902. inline const char *getSeverityTagname(WUExceptionSeverity severity, unsigned flags)
  2903. {
  2904. if (flags & WorkUnitXML_SeverityTags)
  2905. {
  2906. switch (severity)
  2907. {
  2908. case ExceptionSeverityInformation:
  2909. return "Info";
  2910. case ExceptionSeverityWarning:
  2911. return "Warning";
  2912. case ExceptionSeverityAlert:
  2913. return "Alert";
  2914. case ExceptionSeverityError:
  2915. default:
  2916. break;
  2917. }
  2918. }
  2919. return "Exception";
  2920. }
  2921. extern FILEVIEW_API void writeFullWorkUnitResults(const char *username, const char *password, const IConstWorkUnit *cw, IXmlWriter &writer, unsigned flags, WUExceptionSeverity minSeverity, const char *rootTag)
  2922. {
  2923. if (rootTag && *rootTag && !(flags & WorkUnitXML_NoRoot))
  2924. writer.outputBeginNested(rootTag, true);
  2925. Owned<IConstWUExceptionIterator> exceptions = &cw->getExceptions();
  2926. ForEach(*exceptions)
  2927. {
  2928. IConstWUException & exception = exceptions->query();
  2929. WUExceptionSeverity severity = exception.getSeverity();
  2930. if (severity>=minSeverity)
  2931. {
  2932. SCMStringBuffer src, msg, filename;
  2933. exception.getExceptionSource(src);
  2934. exception.getExceptionMessage(msg);
  2935. exception.getExceptionFileName(filename);
  2936. unsigned lineno = exception.getExceptionLineNo();
  2937. unsigned code = exception.getExceptionCode();
  2938. writer.outputBeginNested(getSeverityTagname(severity, flags), false);
  2939. if (code)
  2940. writer.outputUInt(code, "Code");
  2941. if (filename.length())
  2942. writer.outputCString(filename.str(), "Filename");
  2943. if (lineno)
  2944. writer.outputUInt(lineno, "Line");
  2945. writer.outputCString(src.str(), "Source");
  2946. writer.outputCString(msg.str(), "Message");
  2947. writer.outputEndNested(getSeverityTagname(severity, flags));
  2948. }
  2949. }
  2950. Owned<IResultSetFactory> factory = getResultSetFactory(username, password);
  2951. switch (cw->getState())
  2952. {
  2953. case WUStateCompleted:
  2954. case WUStateWait:
  2955. {
  2956. Owned<IConstWUResultIterator> results = &cw->getResults();
  2957. ForEach(*results)
  2958. {
  2959. IConstWUResult &ds = results->query();
  2960. if (ds.getResultSequence()>=0 && (ds.getResultStatus() != ResultStatusUndefined))
  2961. {
  2962. SCMStringBuffer name;
  2963. ds.getResultName(name);
  2964. Owned<INewResultSet> nr = factory->createNewResultSet(&ds, cw->queryWuid());
  2965. const IProperties *xmlns = ds.queryResultXmlns();
  2966. writeResultXml(writer, nr.get(), name.str(), 0, 0, (flags & WorkUnitXML_InclSchema) ? name.str() : NULL, xmlns);
  2967. }
  2968. }
  2969. }
  2970. break;
  2971. case WUStateAborted:
  2972. writer.outputBeginNested("Exception", false);
  2973. writer.outputCString("System", "Source");
  2974. writer.outputCString("Query aborted by operator", "Message");
  2975. writer.outputEndNested("Exception");
  2976. break;
  2977. }
  2978. if (rootTag && *rootTag && !(flags & WorkUnitXML_NoRoot))
  2979. writer.outputEndNested(rootTag);
  2980. }
  2981. extern FILEVIEW_API IStringVal& getFullWorkUnitResultsXML(const char *username, const char *password, const IConstWorkUnit *cw, IStringVal &str, unsigned flags, WUExceptionSeverity minSeverity)
  2982. {
  2983. Owned<CommonXmlWriter> writer = CreateCommonXmlWriter(XWFexpandempty);
  2984. writeFullWorkUnitResults(username, password, cw, *writer, flags, minSeverity, "Result");
  2985. const char *xml = writer->str();
  2986. unsigned len = writer->length();
  2987. if (len && xml[len-1]=='\n')
  2988. len--;
  2989. str.setLen(xml, len);
  2990. return str;
  2991. }
  2992. extern FILEVIEW_API IStringVal& getFullWorkUnitResultsJSON(const char *username, const char *password, const IConstWorkUnit *cw, IStringVal &str, unsigned flags, WUExceptionSeverity minSeverity)
  2993. {
  2994. Owned<CommonJsonWriter> writer = new CommonJsonWriter(0);
  2995. writer->outputBeginRoot();
  2996. writeFullWorkUnitResults(username, password, cw, *writer, flags, minSeverity, "Results");
  2997. writer->outputEndRoot();
  2998. str.set(writer->str());
  2999. return str;
  3000. }