query_expression.hpp 134 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290
  1. /*************************************************************************
  2. *
  3. * Copyright 2016 Realm Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. *
  17. **************************************************************************/
  18. /*
  19. This file lets you write queries in C++ syntax like: Expression* e = (first + 1 / second >= third + 12.3);
  20. Type conversion/promotion semantics is the same as in the C++ expressions, e.g float + int > double == float +
  21. (float)int > double.
  22. Grammar:
  23. -----------------------------------------------------------------------------------------------------------------------
  24. Expression: Subexpr2<T> Compare<Cond, T> Subexpr2<T>
  25. operator! Expression
  26. Subexpr2<T>: Value<T>
  27. Columns<T>
  28. Subexpr2<T> Operator<Oper<T> Subexpr2<T>
  29. power(Subexpr2<T>) // power(x) = x * x, as example of unary operator
  30. Value<T>: T
  31. Operator<Oper<T>>: +, -, *, /
  32. Compare<Cond, T>: ==, !=, >=, <=, >, <
  33. T: bool, int, int64_t, float, double, StringData
  34. Class diagram
  35. -----------------------------------------------------------------------------------------------------------------------
  36. Subexpr2
  37. void evaluate(size_t i, ValueBase* destination)
  38. Compare: public Subexpr2
  39. size_t find_first(size_t start, size_t end) // main method that executes query
  40. unique_ptr<Subexpr2> m_left; // left expression subtree
  41. unique_ptr<Subexpr2> m_right; // right expression subtree
  42. Operator: public Subexpr2
  43. void evaluate(size_t i, ValueBase* destination)
  44. unique_ptr<Subexpr2> m_left; // left expression subtree
  45. unique_ptr<Subexpr2> m_right; // right expression subtree
  46. Value<T>: public Subexpr2
  47. void evaluate(size_t i, ValueBase* destination)
  48. T m_v[8];
  49. Columns<T>: public Subexpr2
  50. void evaluate(size_t i, ValueBase* destination)
  51. SequentialGetter<T> sg; // class bound to a column, lets you read values in a fast way
  52. Table* m_table;
  53. class ColumnAccessor<>: public Columns<double>
  54. Call diagram:
  55. -----------------------------------------------------------------------------------------------------------------------
  56. Example of 'table.first > 34.6 + table.second':
  57. size_t Compare<Greater>::find_first()-------------+
  58. | |
  59. | |
  60. | |
  61. +--> Columns<float>::evaluate() +--------> Operator<Plus>::evaluate()
  62. | |
  63. Value<float>::evaluate() Columns<float>::evaluate()
  64. Operator, Value and Columns have an evaluate(size_t i, ValueBase* destination) method which returns a Value<T>
  65. containing 8 values representing table rows i...i + 7.
  66. So Value<T> contains 8 concecutive values and all operations are based on these chunks. This is
  67. to save overhead by virtual calls needed for evaluating a query that has been dynamically constructed at runtime.
  68. Memory allocation:
  69. -----------------------------------------------------------------------------------------------------------------------
  70. Subexpressions created by the end-user are stack allocated. They are cloned to the heap when passed to UnaryOperator,
  71. Operator, and Compare. Those types own the clones and deallocate them when destroyed.
  72. Caveats, notes and todos
  73. -----------------------------------------------------------------------------------------------------------------------
  74. * Perhaps disallow columns from two different tables in same expression
  75. * The name Columns (with s) an be confusing because we also have Column (without s)
  76. * We have Columns::m_table, Query::m_table and ColumnAccessorBase::m_table that point at the same thing, even with
  77. ColumnAccessor<> extending Columns. So m_table is redundant, but this is in order to keep class dependencies and
  78. entanglement low so that the design is flexible (if you perhaps later want a Columns class that is not dependent
  79. on ColumnAccessor)
  80. Nulls
  81. -----------------------------------------------------------------------------------------------------------------------
  82. First note that at array level, nulls are distinguished between non-null in different ways:
  83. String:
  84. m_data == 0 && m_size == 0
  85. Integer, Bool stored in ArrayIntNull:
  86. value == get(0) (entry 0 determins a magic value that represents nulls)
  87. Float/double:
  88. null::is_null(value) which tests if value bit-matches one specific bit pattern reserved for null
  89. The Columns class encapsulates all this into a simple class that, for any type T has
  90. evaluate(size_t index) that reads values from a column, taking nulls in count
  91. get(index)
  92. set(index)
  93. is_null(index)
  94. set_null(index)
  95. */
  96. #ifndef REALM_QUERY_EXPRESSION_HPP
  97. #define REALM_QUERY_EXPRESSION_HPP
  98. #include <realm/aggregate_ops.hpp>
  99. #include <realm/array_timestamp.hpp>
  100. #include <realm/array_binary.hpp>
  101. #include <realm/array_string.hpp>
  102. #include <realm/array_backlink.hpp>
  103. #include <realm/array_list.hpp>
  104. #include <realm/array_key.hpp>
  105. #include <realm/array_bool.hpp>
  106. #include <realm/array_fixed_bytes.hpp>
  107. #include <realm/column_integer.hpp>
  108. #include <realm/column_type_traits.hpp>
  109. #include <realm/dictionary.hpp>
  110. #include <realm/table.hpp>
  111. #include <realm/index_string.hpp>
  112. #include <realm/query.hpp>
  113. #include <realm/list.hpp>
  114. #include <realm/set.hpp>
  115. #include <realm/query_value.hpp>
  116. #include <realm/metrics/query_info.hpp>
  117. #include <realm/util/optional.hpp>
  118. #include <realm/util/serializer.hpp>
  119. #include <numeric>
  120. #include <algorithm>
  121. // Normally, if a next-generation-syntax condition is supported by the old query_engine.hpp, a query_engine node is
  122. // created because it's faster (by a factor of 5 - 10). Because many of our existing next-generation-syntax unit
  123. // unit tests are indeed simple enough to fallback to old query_engine, query_expression gets low test coverage. Undef
  124. // flag to get higher query_expression test coverage. This is a good idea to try out each time you develop on/modify
  125. // query_expression.
  126. #define REALM_OLDQUERY_FALLBACK 1
  127. namespace realm {
  128. template <class T>
  129. T minimum(T a, T b)
  130. {
  131. return a < b ? a : b;
  132. }
  133. struct Plus {
  134. Mixed operator()(Mixed v1, Mixed v2) const
  135. {
  136. return v1 + v2;
  137. }
  138. static std::string description()
  139. {
  140. return "+";
  141. }
  142. };
  143. struct Minus {
  144. Mixed operator()(Mixed v1, Mixed v2) const
  145. {
  146. return v1 - v2;
  147. }
  148. static std::string description()
  149. {
  150. return "-";
  151. }
  152. };
  153. struct Div {
  154. Mixed operator()(Mixed v1, Mixed v2) const
  155. {
  156. return v1 / v2;
  157. }
  158. static std::string description()
  159. {
  160. return "/";
  161. }
  162. };
  163. struct Mul {
  164. Mixed operator()(Mixed v1, Mixed v2) const
  165. {
  166. return v1 * v2;
  167. }
  168. static std::string description()
  169. {
  170. return "*";
  171. }
  172. };
  173. // Finds a common type for T1 and T2 according to C++ conversion/promotion in arithmetic (float + int => float, etc)
  174. template <class T1, class T2, bool T1_is_int = std::numeric_limits<T1>::is_integer || std::is_same_v<T1, null>,
  175. bool T2_is_int = std::numeric_limits<T2>::is_integer || std::is_same_v<T2, null>,
  176. bool T1_is_widest = (sizeof(T1) > sizeof(T2) || std::is_same_v<T2, null>)>
  177. struct Common;
  178. template <class T1, class T2, bool b>
  179. struct Common<T1, T2, b, b, true> {
  180. typedef T1 type;
  181. };
  182. template <class T1, class T2, bool b>
  183. struct Common<T1, T2, b, b, false> {
  184. typedef T2 type;
  185. };
  186. template <class T1, class T2, bool b>
  187. struct Common<T1, T2, false, true, b> {
  188. typedef T1 type;
  189. };
  190. template <class T1, class T2, bool b>
  191. struct Common<T1, T2, true, false, b> {
  192. typedef T2 type;
  193. };
  194. class ValueBase {
  195. public:
  196. using ValueType = QueryValue;
  197. static const size_t chunk_size = 8;
  198. bool m_from_list = false;
  199. ValueBase() = default;
  200. ValueBase(const ValueType& init_val)
  201. {
  202. m_first[0] = init_val;
  203. }
  204. ~ValueBase()
  205. {
  206. dealloc();
  207. }
  208. ValueBase(const ValueBase& other)
  209. {
  210. *this = other;
  211. }
  212. ValueBase& operator=(const ValueBase& other)
  213. {
  214. m_from_list = other.m_from_list;
  215. set(other.begin(), other.end());
  216. return *this;
  217. }
  218. size_t size() const
  219. {
  220. return m_size;
  221. }
  222. void init(bool from_link_list, size_t nb_values)
  223. {
  224. m_from_list = from_link_list;
  225. m_sorted = false;
  226. resize(nb_values);
  227. }
  228. void init_for_links(bool only_unary_links, size_t size)
  229. {
  230. if (only_unary_links) {
  231. REALM_ASSERT(size <= 1);
  232. init(false, 1);
  233. set_null(0);
  234. }
  235. else {
  236. init(true, size);
  237. }
  238. }
  239. void set_null(size_t ndx)
  240. {
  241. m_first[ndx] = ValueType();
  242. }
  243. template <class T>
  244. void set(size_t ndx, const T& val)
  245. {
  246. if constexpr (std::is_same<T, float>::value || std::is_same<T, double>::value) {
  247. m_first[ndx] = null::is_null_float(val) ? ValueType() : ValueType(val);
  248. }
  249. else {
  250. m_first[ndx] = ValueType(val);
  251. }
  252. }
  253. template <class T>
  254. void set(T b, T e)
  255. {
  256. size_t sz = e - b;
  257. resize(sz);
  258. size_t i = 0;
  259. for (auto from = b; from != e; ++from) {
  260. set(i, *from);
  261. i++;
  262. }
  263. }
  264. ValueType& operator[](size_t n)
  265. {
  266. return m_first[n];
  267. }
  268. const ValueType& operator[](size_t n) const
  269. {
  270. return m_first[n];
  271. }
  272. const ValueType& get(size_t n) const
  273. {
  274. return m_first[n];
  275. }
  276. ValueType* begin()
  277. {
  278. return m_first;
  279. }
  280. const ValueType* begin() const
  281. {
  282. return m_first;
  283. }
  284. ValueType* end()
  285. {
  286. return m_first + m_size;
  287. }
  288. const ValueType* end() const
  289. {
  290. return m_first + m_size;
  291. }
  292. void sort()
  293. {
  294. if (!m_sorted) {
  295. std::sort(begin(), end());
  296. m_sorted = true;
  297. }
  298. }
  299. template <class TOperator>
  300. REALM_FORCEINLINE void fun_const(const ValueType& const_value, const ValueBase& right)
  301. {
  302. TOperator o;
  303. // Operate on values one-by-one
  304. size_t sz = right.size();
  305. init(right.m_from_list, sz);
  306. for (size_t i = 0; i < sz; i++) {
  307. set(i, o(const_value, right[i]));
  308. }
  309. }
  310. template <class TOperator>
  311. REALM_FORCEINLINE void fun_const(const ValueBase& left, const ValueType& const_value)
  312. {
  313. TOperator o;
  314. // Operate on values one-by-one
  315. size_t sz = left.size();
  316. init(left.m_from_list, sz);
  317. for (size_t i = 0; i < sz; i++) {
  318. set(i, o(left[i], const_value));
  319. }
  320. }
  321. template <class TOperator>
  322. REALM_FORCEINLINE void fun(const ValueBase& left, const ValueBase& right)
  323. {
  324. TOperator o;
  325. if (!left.m_from_list && !right.m_from_list) {
  326. // Operate on values one-by-one (one value is one row; no links)
  327. size_t min = std::min(left.size(), right.size());
  328. init(false, min);
  329. for (size_t i = 0; i < min; i++) {
  330. set(i, o(left[i], right[i]));
  331. }
  332. }
  333. else if (left.m_from_list && right.m_from_list) {
  334. // FIXME: Many-to-many links not supported yet. Need to specify behaviour
  335. // Eg: `{1, 2, 3} * {4, 5} > age`
  336. throw LogicError(ErrorCodes::InvalidQuery, "Operations involving two lists are not supported");
  337. }
  338. else if (!left.m_from_list && right.m_from_list) {
  339. // Right values come from link. Left must come from single row.
  340. REALM_ASSERT_DEBUG(left.size() > 0);
  341. init(true, right.size());
  342. auto left_value = left[0];
  343. for (size_t i = 0; i < right.size(); i++) {
  344. set(i, o(left_value, right[i]));
  345. }
  346. }
  347. else if (left.m_from_list && !right.m_from_list) {
  348. // Same as above, but with left values coming from links
  349. REALM_ASSERT_DEBUG(right.size() > 0);
  350. init(true, left.size());
  351. auto right_value = right[0];
  352. for (size_t i = 0; i < left.size(); i++) {
  353. set(i, o(left[i], right_value));
  354. }
  355. }
  356. }
  357. // Given a TCond (==, !=, >, <, >=, <=) and two Value<T>, return index of first match
  358. template <class TCond>
  359. REALM_FORCEINLINE static size_t compare(ValueBase& left, ValueBase& right,
  360. util::Optional<ExpressionComparisonType> left_cmp_type,
  361. util::Optional<ExpressionComparisonType> right_cmp_type)
  362. {
  363. TCond c;
  364. using Compare = ExpressionComparisonType;
  365. if (!left.m_from_list && !right.m_from_list) {
  366. // ALL/NONE not supported for non list types
  367. REALM_ASSERT_DEBUG(!left_cmp_type || *left_cmp_type == Compare::Any);
  368. REALM_ASSERT_DEBUG(!right_cmp_type || *right_cmp_type == Compare::Any);
  369. // Compare values one-by-one (one value is one row; no link lists)
  370. size_t min = minimum(left.size(), right.size());
  371. for (size_t m = 0; m < min; m++) {
  372. if (c(left[m], right[m]))
  373. return m;
  374. }
  375. return not_found;
  376. }
  377. if (left.m_from_list && right.m_from_list && !left_cmp_type && !right_cmp_type) {
  378. // Both lists and no ANY, NONE, ALL specified - simple element by element comparison
  379. if (left.size() != right.size()) {
  380. if constexpr (std::is_same_v<TCond, NotEqual>) {
  381. return 0; // mismatch size
  382. }
  383. else {
  384. return not_found;
  385. }
  386. }
  387. for (size_t i = 0; i < left.size(); ++i) {
  388. if (!c(left[i], right[i])) {
  389. return not_found;
  390. }
  391. }
  392. return 0; // all elements matched in the right order
  393. }
  394. // if one side omitted a comparison type, assume ANY
  395. const Compare compare_left = left_cmp_type.value_or(Compare::Any);
  396. const Compare compare_right = right_cmp_type.value_or(Compare::Any);
  397. size_t left_size = left.m_from_list ? left.size() : 1;
  398. size_t right_size = right.m_from_list ? right.size() : 1;
  399. if (left_size > 2 && right_size > 2) {
  400. left.sort();
  401. right.sort();
  402. if constexpr (std::is_same_v<TCond, Equal>) {
  403. if (compare_left != ExpressionComparisonType::None && compare_right == Compare::Any) {
  404. // Optimization with O(n) complexity
  405. const bool any = compare_left == ExpressionComparisonType::Any;
  406. size_t left_idx = 0;
  407. size_t right_idx = 0;
  408. while (right_idx < right_size) {
  409. if (c(left[left_idx], right[right_idx])) {
  410. left_idx++;
  411. right_idx++;
  412. if (any || left_idx == left_size) {
  413. return 0;
  414. }
  415. }
  416. else {
  417. if (left[left_idx] < right[right_idx]) {
  418. if (any && left_idx < left_size) {
  419. left_idx++;
  420. }
  421. else {
  422. return not_found;
  423. }
  424. }
  425. else {
  426. right_idx++;
  427. }
  428. }
  429. }
  430. return not_found;
  431. }
  432. }
  433. else if constexpr (realm::is_any_v<TCond, Greater, GreaterEqual, Less, LessEqual>) {
  434. // Only consider first and last
  435. left[1] = left[left_size - 1];
  436. left_size = 2;
  437. right[1] = right[right_size - 1];
  438. right_size = 2;
  439. }
  440. else {
  441. // remove duplicates to reduce comparison time in nested loops
  442. left_size = std::unique(left.begin(), left.end()) - left.begin();
  443. right_size = std::unique(right.begin(), right.end()) - right.begin();
  444. }
  445. }
  446. if constexpr (realm::is_any_v<TCond, BeginsWith, BeginsWithIns, EndsWith, EndsWithIns, Contains, ContainsIns,
  447. Like, LikeIns>) {
  448. // The string operators have the arguments reversed so we have to iterate right in the
  449. // outer loop as this is actually the left argument
  450. auto left_matches = [&](const QueryValue& right_val) {
  451. for (size_t i = 0; i < left_size; i++) {
  452. if (c(left[i], right_val)) {
  453. // match
  454. if (compare_left == Compare::Any) {
  455. return true;
  456. }
  457. if (compare_left == Compare::None) {
  458. return false; // one matched
  459. }
  460. }
  461. else {
  462. // no match
  463. if (compare_left == Compare::All) {
  464. return false;
  465. }
  466. }
  467. }
  468. if (compare_left == Compare::None || compare_left == Compare::All) {
  469. return true;
  470. }
  471. return false;
  472. };
  473. for (size_t i = 0; i < right_size; i++) {
  474. if (left_matches(right[i])) {
  475. if (compare_right == Compare::Any) {
  476. return 0;
  477. }
  478. if (compare_right == Compare::None) {
  479. return not_found; // one matched
  480. }
  481. }
  482. else {
  483. if (compare_right == Compare::All) {
  484. return not_found;
  485. }
  486. }
  487. }
  488. if (compare_right == Compare::None || compare_right == Compare::All) {
  489. return 0; // either none or all
  490. }
  491. }
  492. else {
  493. auto right_matches = [&](const QueryValue& left_val) {
  494. for (size_t i = 0; i < right_size; i++) {
  495. if (c(left_val, right[i])) {
  496. // match
  497. if (compare_right == Compare::Any) {
  498. return true;
  499. }
  500. if (compare_right == Compare::None) {
  501. return false; // one matched
  502. }
  503. }
  504. else {
  505. // no match
  506. if (compare_right == Compare::All) {
  507. return false;
  508. }
  509. }
  510. }
  511. if (compare_right == Compare::None || compare_right == Compare::All) {
  512. return true;
  513. }
  514. return false;
  515. };
  516. for (size_t i = 0; i < left_size; i++) {
  517. if (right_matches(left[i])) {
  518. if (compare_left == Compare::Any) {
  519. return 0;
  520. }
  521. if (compare_left == ExpressionComparisonType::None) {
  522. return not_found; // one matched
  523. }
  524. }
  525. else {
  526. if (compare_left == ExpressionComparisonType::All) {
  527. return not_found;
  528. }
  529. }
  530. }
  531. if (compare_left == ExpressionComparisonType::None || compare_left == ExpressionComparisonType::All) {
  532. return 0; // either none or all
  533. }
  534. }
  535. return not_found; // no match
  536. }
  537. private:
  538. // If true, all values in the class come from a link list of a single field in the parent table (m_table). If
  539. // false, then values come from successive rows of m_table (query operations are operated on in bulks for speed)
  540. static constexpr size_t prealloc = 8;
  541. QueryValue m_cache[prealloc];
  542. QueryValue* m_first = &m_cache[0];
  543. size_t m_size = 1;
  544. bool m_sorted = false;
  545. void resize(size_t size)
  546. {
  547. if (size == m_size)
  548. return;
  549. dealloc();
  550. m_size = size;
  551. if (m_size > 0) {
  552. if (m_size > prealloc)
  553. m_first = new QueryValue[m_size];
  554. else
  555. m_first = &m_cache[0];
  556. }
  557. }
  558. void dealloc()
  559. {
  560. if (m_first) {
  561. if (m_size > prealloc)
  562. delete[] m_first;
  563. m_first = nullptr;
  564. }
  565. }
  566. void fill(const QueryValue& val)
  567. {
  568. for (size_t i = 0; i < m_size; i++) {
  569. m_first[i] = val;
  570. }
  571. }
  572. };
  573. class Expression {
  574. public:
  575. Expression() {}
  576. virtual ~Expression() {}
  577. virtual double init()
  578. {
  579. return 50.0; // Default dT
  580. }
  581. virtual size_t find_first(size_t start, size_t end) const = 0;
  582. virtual void set_base_table(ConstTableRef table) = 0;
  583. virtual void set_cluster(const Cluster*) = 0;
  584. virtual void collect_dependencies(std::vector<TableKey>&) const {}
  585. virtual ConstTableRef get_base_table() const = 0;
  586. virtual std::string description(util::serializer::SerialisationState& state) const = 0;
  587. virtual std::unique_ptr<Expression> clone() const = 0;
  588. };
  589. template <typename T, typename... Args>
  590. std::unique_ptr<Expression> make_expression(Args&&... args)
  591. {
  592. return std::unique_ptr<Expression>(new T(std::forward<Args>(args)...));
  593. }
  594. class Subexpr {
  595. public:
  596. virtual ~Subexpr() {}
  597. virtual std::unique_ptr<Subexpr> clone() const = 0;
  598. // When the user constructs a query, it always "belongs" to one single base/parent table (regardless of
  599. // any links or not and regardless of any queries assembled with || or &&). When you do a Query::find(),
  600. // then Query::m_table is set to this table, and set_base_table() is called on all Columns and LinkMaps in
  601. // the query expression tree so that they can set/update their internals as required.
  602. //
  603. // During thread-handover of a Query, set_base_table() is also called to make objects point at the new table
  604. // instead of the old one from the old thread.
  605. virtual void set_base_table(ConstTableRef) {}
  606. virtual std::string description(util::serializer::SerialisationState& state) const = 0;
  607. virtual void set_cluster(const Cluster*) {}
  608. // Recursively fetch tables of columns in expression tree. Used when user first builds a stand-alone expression
  609. // and
  610. // binds it to a Query at a later time
  611. virtual ConstTableRef get_base_table() const
  612. {
  613. return nullptr;
  614. }
  615. virtual void collect_dependencies(std::vector<TableKey>&) const {}
  616. virtual bool has_constant_evaluation() const
  617. {
  618. return false;
  619. }
  620. virtual bool has_single_value() const
  621. {
  622. return false;
  623. }
  624. virtual bool has_multiple_values() const
  625. {
  626. return false;
  627. }
  628. virtual bool has_search_index() const
  629. {
  630. return false;
  631. }
  632. virtual std::vector<ObjKey> find_all(Mixed) const
  633. {
  634. return {};
  635. }
  636. virtual ConstTableRef get_target_table() const
  637. {
  638. return {};
  639. }
  640. virtual DataType get_type() const = 0;
  641. virtual void evaluate(size_t index, ValueBase& destination) = 0;
  642. // This function supports SubColumnAggregate
  643. virtual void evaluate(ObjKey, ValueBase&)
  644. {
  645. REALM_ASSERT(false); // Unimplemented
  646. }
  647. virtual Mixed get_mixed()
  648. {
  649. return {};
  650. }
  651. virtual util::Optional<ExpressionComparisonType> get_comparison_type() const
  652. {
  653. return util::none;
  654. }
  655. };
  656. template <typename T, typename... Args>
  657. std::unique_ptr<Subexpr> make_subexpr(Args&&... args)
  658. {
  659. return std::unique_ptr<Subexpr>(new T(std::forward<Args>(args)...));
  660. }
  661. template <class T>
  662. class Columns;
  663. template <class T>
  664. class Value;
  665. class ConstantMixedValue;
  666. template <class T>
  667. class Subexpr2;
  668. template <class oper>
  669. class Operator;
  670. template <class oper, class TLeft = Subexpr>
  671. class UnaryOperator;
  672. template <class oper, class TLeft = Subexpr>
  673. class SizeOperator;
  674. template <class oper>
  675. class TypeOfValueOperator;
  676. template <class TCond>
  677. class Compare;
  678. template <bool has_links>
  679. class UnaryLinkCompare;
  680. class ColumnAccessorBase;
  681. // Handle cases where left side is a constant (int, float, int64_t, double, StringData)
  682. template <class Cond, class L, class R>
  683. Query create(L left, const Subexpr2<R>& right)
  684. {
  685. // Purpose of below code is to intercept the creation of a condition and test if it's supported by the old
  686. // query_engine.hpp which is faster. If it's supported, create a query_engine.hpp node, otherwise create a
  687. // query_expression.hpp node.
  688. //
  689. // This method intercepts only Value <cond> Subexpr2. Interception of Subexpr2 <cond> Subexpr is elsewhere.
  690. constexpr const bool supported_by_old_query_engine =
  691. (std::numeric_limits<L>::is_integer && std::numeric_limits<R>::is_integer) || std::is_same_v<R, Mixed> ||
  692. (std::is_same_v<L, R> &&
  693. realm::is_any_v<L, double, float, Timestamp, StringData, BinaryData, ObjectId, UUID>);
  694. if constexpr (REALM_OLDQUERY_FALLBACK && supported_by_old_query_engine) {
  695. const Columns<R>* column = dynamic_cast<const Columns<R>*>(&right);
  696. // TODO: recognize size operator expressions
  697. // auto size_operator = dynamic_cast<const SizeOperator<Size<StringData>, Subexpr>*>(&right);
  698. if (column && !column->links_exist()) {
  699. ConstTableRef t = column->get_base_table();
  700. Query q(t);
  701. if constexpr (std::is_same_v<Cond, Less>)
  702. q.greater(column->column_key(), static_cast<R>(left));
  703. else if constexpr (std::is_same_v<Cond, Greater>)
  704. q.less(column->column_key(), static_cast<R>(left));
  705. else if constexpr (std::is_same_v<Cond, Equal>)
  706. q.equal(column->column_key(), static_cast<R>(left));
  707. else if constexpr (std::is_same_v<Cond, NotEqual>)
  708. q.not_equal(column->column_key(), static_cast<R>(left));
  709. else if constexpr (std::is_same_v<Cond, LessEqual>)
  710. q.greater_equal(column->column_key(), static_cast<R>(left));
  711. else if constexpr (std::is_same_v<Cond, GreaterEqual>)
  712. q.less_equal(column->column_key(), static_cast<R>(left));
  713. else if constexpr (std::is_same_v<Cond, EqualIns>)
  714. q.equal(column->column_key(), left, false);
  715. else if constexpr (std::is_same_v<Cond, NotEqualIns>)
  716. q.not_equal(column->column_key(), left, false);
  717. else if constexpr (std::is_same_v<Cond, BeginsWith>)
  718. q.begins_with(column->column_key(), left);
  719. else if constexpr (std::is_same_v<Cond, BeginsWithIns>)
  720. q.begins_with(column->column_key(), left, false);
  721. else if constexpr (std::is_same_v<Cond, EndsWith>)
  722. q.ends_with(column->column_key(), left);
  723. else if constexpr (std::is_same_v<Cond, EndsWithIns>)
  724. q.ends_with(column->column_key(), left, false);
  725. else if constexpr (std::is_same_v<Cond, Contains>)
  726. q.contains(column->column_key(), left);
  727. else if constexpr (std::is_same_v<Cond, ContainsIns>)
  728. q.contains(column->column_key(), left, false);
  729. else if constexpr (std::is_same_v<Cond, Like>)
  730. q.like(column->column_key(), left);
  731. else if constexpr (std::is_same_v<Cond, LikeIns>)
  732. q.like(column->column_key(), left, false);
  733. else {
  734. // query_engine.hpp does not support this Cond. Please either add support for it in query_engine.hpp
  735. // or fallback to using use 'return new Compare<>' instead.
  736. REALM_ASSERT(false);
  737. }
  738. return q;
  739. }
  740. }
  741. // Return query_expression.hpp node
  742. if constexpr (std::is_same_v<L, TypeOfValue>) {
  743. return make_expression<Compare<Cond>>(make_subexpr<Value<TypeOfValue>>(left), right.clone());
  744. }
  745. else {
  746. return make_expression<Compare<Cond>>(make_subexpr<ConstantMixedValue>(left), right.clone());
  747. }
  748. }
  749. // Purpose of this method is to intercept the creation of a condition and test if it's supported by the old
  750. // query_engine.hpp which is faster. If it's supported, create a query_engine.hpp node, otherwise create a
  751. // query_expression.hpp node.
  752. //
  753. // This method intercepts Subexpr2 <cond> Subexpr2 only. Value <cond> Subexpr2 is intercepted elsewhere.
  754. template <class Cond, typename L, typename R>
  755. Query create2(const Subexpr2<L>& left, const Subexpr2<R>& right)
  756. {
  757. #ifdef REALM_OLDQUERY_FALLBACK // if not defined, never fallback query_engine; always use query_expression
  758. // Test if expressions are of type Columns. Other possibilities are Value and Operator.
  759. const Columns<L>* left_col = dynamic_cast<const Columns<L>*>(&left);
  760. const Columns<R>* right_col = dynamic_cast<const Columns<R>*>(&right);
  761. // query_engine supports 'T-column <op> <T-column>' for T = {int64_t, float, double}, op = {<, >, ==, !=, <=,
  762. // >=},
  763. // but only if both columns are non-nullable, and aren't in linked tables.
  764. if (left_col && right_col) {
  765. ConstTableRef t = left_col->get_base_table();
  766. ConstTableRef t_right = right_col->get_base_table();
  767. REALM_ASSERT_DEBUG(t);
  768. REALM_ASSERT_DEBUG(t_right);
  769. // we only support multi column comparisons if they stem from the same table
  770. if (t->get_key() != t_right->get_key()) {
  771. throw Exception(
  772. ErrorCodes::InvalidQuery,
  773. util::format(
  774. "Comparison between two properties must be linked with a relationship or exist on the same "
  775. "Table (%1 and %2)",
  776. t->get_name(), t_right->get_name()));
  777. }
  778. if (!left_col->links_exist() && !right_col->links_exist()) {
  779. if constexpr (std::is_same_v<Cond, Less>)
  780. return Query(t).less(left_col->column_key(), right_col->column_key());
  781. if constexpr (std::is_same_v<Cond, Greater>)
  782. return Query(t).greater(left_col->column_key(), right_col->column_key());
  783. if constexpr (std::is_same_v<Cond, Equal>)
  784. return Query(t).equal(left_col->column_key(), right_col->column_key());
  785. if constexpr (std::is_same_v<Cond, NotEqual>)
  786. return Query(t).not_equal(left_col->column_key(), right_col->column_key());
  787. if constexpr (std::is_same_v<Cond, LessEqual>)
  788. return Query(t).less_equal(left_col->column_key(), right_col->column_key());
  789. if constexpr (std::is_same_v<Cond, GreaterEqual>)
  790. return Query(t).greater_equal(left_col->column_key(), right_col->column_key());
  791. }
  792. }
  793. #endif
  794. // Return query_expression.hpp node
  795. return make_expression<Compare<Cond>>(left.clone(), right.clone());
  796. }
  797. // All overloads where left-hand-side is Subexpr2<L>:
  798. //
  799. // left-hand-side operator right-hand-side
  800. // Subexpr2<L> +, -, *, /, <, >, ==, !=, <=, >= R, Subexpr2<R>
  801. //
  802. // For L = R = {int, int64_t, float, double, StringData, Timestamp}:
  803. template <class L, class R>
  804. class Overloads {
  805. public:
  806. // Compare, right side constant
  807. friend Query operator>(const Subexpr2<L>& left, R right)
  808. {
  809. return create<Less>(right, left);
  810. }
  811. friend Query operator<(const Subexpr2<L>& left, R right)
  812. {
  813. return create<Greater>(right, left);
  814. }
  815. friend Query operator>=(const Subexpr2<L>& left, R right)
  816. {
  817. return create<LessEqual>(right, left);
  818. }
  819. friend Query operator<=(const Subexpr2<L>& left, R right)
  820. {
  821. return create<GreaterEqual>(right, left);
  822. }
  823. friend Query operator==(const Subexpr2<L>& left, R right)
  824. {
  825. return create<Equal>(right, left);
  826. }
  827. friend Query operator!=(const Subexpr2<L>& left, R right)
  828. {
  829. return create<NotEqual>(right, left);
  830. }
  831. // Compare left-side constant
  832. friend Query operator>(R left, const Subexpr2<L>& right)
  833. {
  834. return create<Greater>(left, right);
  835. }
  836. friend Query operator<(R left, const Subexpr2<L>& right)
  837. {
  838. return create<Less>(left, right);
  839. }
  840. friend Query operator>=(R left, const Subexpr2<L>& right)
  841. {
  842. return create<GreaterEqual>(left, right);
  843. }
  844. friend Query operator<=(R left, const Subexpr2<L>& right)
  845. {
  846. return create<LessEqual>(left, right);
  847. }
  848. friend Query operator==(R left, const Subexpr2<L>& right)
  849. {
  850. return create<Equal>(left, right);
  851. }
  852. friend Query operator!=(R left, const Subexpr2<L>& right)
  853. {
  854. return create<NotEqual>(left, right);
  855. }
  856. // Compare, right side subexpression
  857. friend Query operator==(const Subexpr2<L>& left, const Subexpr2<R>& right)
  858. {
  859. return create2<Equal>(left, right);
  860. }
  861. friend Query operator!=(const Subexpr2<L>& left, const Subexpr2<R>& right)
  862. {
  863. return create2<NotEqual>(left, right);
  864. }
  865. friend Query operator>(const Subexpr2<L>& left, const Subexpr2<R>& right)
  866. {
  867. return create2<Greater>(left, right);
  868. }
  869. friend Query operator<(const Subexpr2<L>& left, const Subexpr2<R>& right)
  870. {
  871. return create2<Less>(left, right);
  872. }
  873. friend Query operator>=(const Subexpr2<L>& left, const Subexpr2<R>& right)
  874. {
  875. return create2<GreaterEqual>(left, right);
  876. }
  877. friend Query operator<=(const Subexpr2<L>& left, const Subexpr2<R>& right)
  878. {
  879. return create2<LessEqual>(left, right);
  880. }
  881. };
  882. // With this wrapper class we can define just 20 overloads inside Overloads<L, R> instead of 5 * 20 = 100. Todo: We
  883. // can
  884. // consider if it's simpler/better to remove this class completely and just list all 100 overloads manually anyway.
  885. template <class T>
  886. class Subexpr2 : public Subexpr,
  887. public Overloads<T, int>,
  888. public Overloads<T, float>,
  889. public Overloads<T, double>,
  890. public Overloads<T, int64_t>,
  891. public Overloads<T, StringData>,
  892. public Overloads<T, bool>,
  893. public Overloads<T, Timestamp>,
  894. public Overloads<T, ObjectId>,
  895. public Overloads<T, Decimal128>,
  896. public Overloads<T, UUID>,
  897. public Overloads<T, Mixed>,
  898. public Overloads<T, null> {
  899. public:
  900. DataType get_type() const override
  901. {
  902. return ColumnTypeTraits<T>::id;
  903. }
  904. };
  905. template <class Operator>
  906. Query compare(const Subexpr2<Link>& left, const Obj& obj);
  907. template <class Operator>
  908. Query compare(const Subexpr2<Link>& left, null obj);
  909. // Subexpr2<Link> only provides equality comparisons. Their implementations can be found later in this file.
  910. template <>
  911. class Subexpr2<Link> : public Subexpr {
  912. public:
  913. DataType get_type() const
  914. {
  915. return type_Link;
  916. }
  917. friend Query operator==(const Subexpr2<Link>& left, const Obj& row)
  918. {
  919. return compare<Equal>(left, row);
  920. }
  921. friend Query operator!=(const Subexpr2<Link>& left, const Obj& row)
  922. {
  923. return compare<NotEqual>(left, row);
  924. }
  925. friend Query operator==(const Obj& row, const Subexpr2<Link>& right)
  926. {
  927. return compare<Equal>(right, row);
  928. }
  929. friend Query operator!=(const Obj& row, const Subexpr2<Link>& right)
  930. {
  931. return compare<NotEqual>(right, row);
  932. }
  933. friend Query operator==(const Subexpr2<Link>& left, null)
  934. {
  935. return compare<Equal>(left, null());
  936. }
  937. friend Query operator!=(const Subexpr2<Link>& left, null)
  938. {
  939. return compare<NotEqual>(left, null());
  940. }
  941. friend Query operator==(null, const Subexpr2<Link>& right)
  942. {
  943. return compare<Equal>(right, null());
  944. }
  945. friend Query operator!=(null, const Subexpr2<Link>& right)
  946. {
  947. return compare<NotEqual>(right, null());
  948. }
  949. friend Query operator==(const Subexpr2<Link>& left, const Subexpr2<Link>& right)
  950. {
  951. return make_expression<Compare<Equal>>(left.clone(), right.clone());
  952. }
  953. friend Query operator!=(const Subexpr2<Link>& left, const Subexpr2<Link>& right)
  954. {
  955. return make_expression<Compare<NotEqual>>(left.clone(), right.clone());
  956. }
  957. };
  958. template <>
  959. class Subexpr2<StringData> : public Subexpr, public Overloads<StringData, StringData> {
  960. public:
  961. Query equal(StringData sd, bool case_sensitive = true);
  962. Query equal(const Subexpr2<StringData>& col, bool case_sensitive = true);
  963. Query not_equal(StringData sd, bool case_sensitive = true);
  964. Query not_equal(const Subexpr2<StringData>& col, bool case_sensitive = true);
  965. Query begins_with(StringData sd, bool case_sensitive = true);
  966. Query begins_with(const Subexpr2<StringData>& col, bool case_sensitive = true);
  967. Query ends_with(StringData sd, bool case_sensitive = true);
  968. Query ends_with(const Subexpr2<StringData>& col, bool case_sensitive = true);
  969. Query contains(StringData sd, bool case_sensitive = true);
  970. Query contains(const Subexpr2<StringData>& col, bool case_sensitive = true);
  971. Query like(StringData sd, bool case_sensitive = true);
  972. Query like(const Subexpr2<StringData>& col, bool case_sensitive = true);
  973. DataType get_type() const final
  974. {
  975. return type_String;
  976. }
  977. };
  978. template <>
  979. class Subexpr2<BinaryData> : public Subexpr, public Overloads<BinaryData, BinaryData> {
  980. public:
  981. Query equal(BinaryData sd, bool case_sensitive = true);
  982. Query equal(const Subexpr2<BinaryData>& col, bool case_sensitive = true);
  983. Query not_equal(BinaryData sd, bool case_sensitive = true);
  984. Query not_equal(const Subexpr2<BinaryData>& col, bool case_sensitive = true);
  985. Query begins_with(BinaryData sd, bool case_sensitive = true);
  986. Query begins_with(const Subexpr2<BinaryData>& col, bool case_sensitive = true);
  987. Query ends_with(BinaryData sd, bool case_sensitive = true);
  988. Query ends_with(const Subexpr2<BinaryData>& col, bool case_sensitive = true);
  989. Query contains(BinaryData sd, bool case_sensitive = true);
  990. Query contains(const Subexpr2<BinaryData>& col, bool case_sensitive = true);
  991. Query like(BinaryData sd, bool case_sensitive = true);
  992. Query like(const Subexpr2<BinaryData>& col, bool case_sensitive = true);
  993. DataType get_type() const final
  994. {
  995. return type_Binary;
  996. }
  997. };
  998. template <>
  999. class Subexpr2<Mixed> : public Subexpr,
  1000. public Overloads<Mixed, Mixed>,
  1001. public Overloads<Mixed, int>,
  1002. public Overloads<Mixed, float>,
  1003. public Overloads<Mixed, double>,
  1004. public Overloads<Mixed, int64_t>,
  1005. public Overloads<Mixed, StringData>,
  1006. public Overloads<Mixed, bool>,
  1007. public Overloads<Mixed, Timestamp>,
  1008. public Overloads<Mixed, ObjectId>,
  1009. public Overloads<Mixed, Decimal128>,
  1010. public Overloads<Mixed, UUID>,
  1011. public Overloads<Mixed, null> {
  1012. public:
  1013. Query equal(Mixed sd, bool case_sensitive = true);
  1014. Query equal(const Subexpr2<Mixed>& col, bool case_sensitive = true);
  1015. Query not_equal(Mixed sd, bool case_sensitive = true);
  1016. Query not_equal(const Subexpr2<Mixed>& col, bool case_sensitive = true);
  1017. Query begins_with(Mixed sd, bool case_sensitive = true);
  1018. Query begins_with(const Subexpr2<Mixed>& col, bool case_sensitive = true);
  1019. Query ends_with(Mixed sd, bool case_sensitive = true);
  1020. Query ends_with(const Subexpr2<Mixed>& col, bool case_sensitive = true);
  1021. Query contains(Mixed sd, bool case_sensitive = true);
  1022. Query contains(const Subexpr2<Mixed>& col, bool case_sensitive = true);
  1023. Query like(Mixed sd, bool case_sensitive = true);
  1024. Query like(const Subexpr2<Mixed>& col, bool case_sensitive = true);
  1025. DataType get_type() const override
  1026. {
  1027. return type_Mixed;
  1028. }
  1029. using T = Mixed; // used inside the following macros for operator overloads
  1030. };
  1031. template <>
  1032. class Subexpr2<TypeOfValue> : public Subexpr, public Overloads<TypeOfValue, TypeOfValue> {
  1033. public:
  1034. Query equal(TypeOfValue v);
  1035. Query equal(const TypeOfValueOperator<Mixed>& col);
  1036. Query not_equal(TypeOfValue v);
  1037. Query not_equal(const TypeOfValueOperator<Mixed>& col);
  1038. DataType get_type() const final
  1039. {
  1040. return type_TypeOfValue;
  1041. }
  1042. };
  1043. struct TrueExpression : Expression {
  1044. size_t find_first(size_t start, size_t end) const override
  1045. {
  1046. REALM_ASSERT(start <= end);
  1047. if (start != end)
  1048. return start;
  1049. return realm::not_found;
  1050. }
  1051. void set_base_table(ConstTableRef) override {}
  1052. void set_cluster(const Cluster*) override {}
  1053. ConstTableRef get_base_table() const override
  1054. {
  1055. return nullptr;
  1056. }
  1057. std::string description(util::serializer::SerialisationState&) const override
  1058. {
  1059. return "TRUEPREDICATE";
  1060. }
  1061. std::unique_ptr<Expression> clone() const override
  1062. {
  1063. return std::unique_ptr<Expression>(new TrueExpression(*this));
  1064. }
  1065. };
  1066. struct FalseExpression : Expression {
  1067. size_t find_first(size_t, size_t) const override
  1068. {
  1069. return realm::not_found;
  1070. }
  1071. void set_base_table(ConstTableRef) override {}
  1072. void set_cluster(const Cluster*) override {}
  1073. std::string description(util::serializer::SerialisationState&) const override
  1074. {
  1075. return "FALSEPREDICATE";
  1076. }
  1077. ConstTableRef get_base_table() const override
  1078. {
  1079. return nullptr;
  1080. }
  1081. std::unique_ptr<Expression> clone() const override
  1082. {
  1083. return std::unique_ptr<Expression>(new FalseExpression(*this));
  1084. }
  1085. };
  1086. // Stores N values of type T. Can also exchange data with other ValueBase of different types
  1087. template <class T>
  1088. class Value : public ValueBase, public Subexpr2<T> {
  1089. public:
  1090. Value() = default;
  1091. Value(T init)
  1092. : ValueBase(QueryValue(init))
  1093. {
  1094. }
  1095. std::string value_to_string(size_t ndx, util::serializer::SerialisationState& state) const
  1096. {
  1097. auto val = get(ndx);
  1098. if (val.is_null())
  1099. return "NULL";
  1100. else {
  1101. static_cast<void>(state);
  1102. if constexpr (std::is_same_v<T, TypeOfValue>) {
  1103. return util::serializer::print_value(val.get_type_of_value());
  1104. }
  1105. else if constexpr (std::is_same_v<T, ObjKey>) {
  1106. ObjKey obj_key = val.template get<ObjKey>();
  1107. if (state.target_table) {
  1108. ObjLink link(state.target_table->get_key(), obj_key);
  1109. return util::serializer::print_value(link, state.group);
  1110. }
  1111. else {
  1112. return util::serializer::print_value(obj_key);
  1113. }
  1114. }
  1115. else if constexpr (std::is_same_v<T, ObjLink>) {
  1116. return util::serializer::print_value(val.template get<ObjLink>(), state.group);
  1117. }
  1118. else if constexpr (std::is_same_v<T, Mixed>) {
  1119. if (val.is_type(type_TypedLink)) {
  1120. return util::serializer::print_value(val.template get<ObjLink>(), state.group);
  1121. }
  1122. else {
  1123. return util::serializer::print_value(val);
  1124. }
  1125. }
  1126. else {
  1127. return util::serializer::print_value(val.template get<T>());
  1128. }
  1129. }
  1130. }
  1131. std::string description(util::serializer::SerialisationState& state) const override
  1132. {
  1133. const size_t sz = size();
  1134. if (m_from_list) {
  1135. std::string desc = state.describe_expression_type(m_comparison_type) + "{";
  1136. for (size_t i = 0; i < sz; ++i) {
  1137. if (i != 0) {
  1138. desc += ", ";
  1139. }
  1140. desc += value_to_string(i, state);
  1141. }
  1142. desc += "}";
  1143. return desc;
  1144. }
  1145. else if (sz == 1) {
  1146. return value_to_string(0, state);
  1147. }
  1148. return "";
  1149. }
  1150. bool has_multiple_values() const override
  1151. {
  1152. return m_from_list;
  1153. }
  1154. bool has_single_value() const override
  1155. {
  1156. return size() == 1;
  1157. }
  1158. bool has_constant_evaluation() const override
  1159. {
  1160. return true;
  1161. }
  1162. util::Optional<ExpressionComparisonType> get_comparison_type() const final
  1163. {
  1164. REALM_ASSERT_DEBUG(!m_comparison_type || m_from_list);
  1165. return m_comparison_type;
  1166. }
  1167. void set_comparison_type(util::Optional<ExpressionComparisonType> type)
  1168. {
  1169. m_comparison_type = type;
  1170. }
  1171. Mixed get_mixed() override
  1172. {
  1173. return get(0);
  1174. }
  1175. void evaluate(size_t, ValueBase& destination) override
  1176. {
  1177. destination = *this;
  1178. }
  1179. std::unique_ptr<Subexpr> clone() const override
  1180. {
  1181. return make_subexpr<Value<T>>(*this);
  1182. }
  1183. protected:
  1184. util::Optional<ExpressionComparisonType> m_comparison_type;
  1185. };
  1186. class ConstantMixedValue : public Value<Mixed> {
  1187. public:
  1188. ConstantMixedValue(const Mixed& val)
  1189. : Value(val)
  1190. {
  1191. begin()->use_buffer(m_buffer);
  1192. }
  1193. std::unique_ptr<Subexpr> clone() const override
  1194. {
  1195. return std::unique_ptr<Subexpr>(new ConstantMixedValue(*this));
  1196. }
  1197. private:
  1198. ConstantMixedValue(const ConstantMixedValue& other)
  1199. : Value(other)
  1200. {
  1201. begin()->use_buffer(m_buffer);
  1202. }
  1203. std::string m_buffer;
  1204. };
  1205. class ConstantMixedList : public Value<Mixed> {
  1206. public:
  1207. ConstantMixedList(size_t nb_values)
  1208. : Value()
  1209. , m_buffer(nb_values)
  1210. {
  1211. this->init(true, nb_values);
  1212. }
  1213. void set(size_t n, Mixed val)
  1214. {
  1215. Value<Mixed>::set(n, val);
  1216. (*this)[n].use_buffer(m_buffer[n]);
  1217. }
  1218. std::unique_ptr<Subexpr> clone() const override
  1219. {
  1220. return std::unique_ptr<Subexpr>(new ConstantMixedList(*this));
  1221. }
  1222. private:
  1223. ConstantMixedList(const ConstantMixedList& other)
  1224. : Value(other)
  1225. , m_buffer(other.size())
  1226. {
  1227. for (size_t i = 0; i < size(); i++) {
  1228. (*this)[i].use_buffer(m_buffer[i]);
  1229. }
  1230. }
  1231. std::vector<std::string> m_buffer;
  1232. };
  1233. class ConstantStringValue : public Value<StringData> {
  1234. public:
  1235. ConstantStringValue(const StringData& string)
  1236. : Value()
  1237. , m_string(string.is_null() ? util::none : util::make_optional(std::string(string)))
  1238. {
  1239. if (m_string)
  1240. set(0, *m_string);
  1241. }
  1242. std::unique_ptr<Subexpr> clone() const override
  1243. {
  1244. return std::unique_ptr<Subexpr>(new ConstantStringValue(*this));
  1245. }
  1246. private:
  1247. ConstantStringValue(const ConstantStringValue& other)
  1248. : Value()
  1249. , m_string(other.m_string)
  1250. {
  1251. if (m_string)
  1252. set(0, *m_string);
  1253. }
  1254. util::Optional<std::string> m_string;
  1255. };
  1256. class ConstantBinaryValue : public Value<BinaryData> {
  1257. public:
  1258. ConstantBinaryValue(const BinaryData& bin)
  1259. : Value()
  1260. , m_buffer(bin)
  1261. {
  1262. if (m_buffer.data())
  1263. set(0, BinaryData(m_buffer.data(), m_buffer.size()));
  1264. }
  1265. std::unique_ptr<Subexpr> clone() const override
  1266. {
  1267. return std::unique_ptr<Subexpr>(new ConstantBinaryValue(*this));
  1268. }
  1269. private:
  1270. ConstantBinaryValue(const ConstantBinaryValue& other)
  1271. : Value()
  1272. , m_buffer(other.m_buffer)
  1273. {
  1274. if (m_buffer.data())
  1275. set(0, BinaryData(m_buffer.data(), m_buffer.size()));
  1276. }
  1277. OwnedBinaryData m_buffer;
  1278. };
  1279. // Classes used for LinkMap (see below).
  1280. struct LinkMapFunction {
  1281. // Your consume() method is given key within the linked-to table as argument, and you must return whether or
  1282. // not you want the LinkMapFunction to exit (return false) or continue (return true) harvesting the link tree
  1283. // for the current main table object (it will be a link tree if you have multiple type_LinkList columns
  1284. // in a link()->link() query.
  1285. virtual bool consume(ObjKey) = 0;
  1286. };
  1287. struct FindNullLinks : public LinkMapFunction {
  1288. bool consume(ObjKey) override
  1289. {
  1290. m_has_link = true;
  1291. return false; // we've found a key, so this can't be a null-link, so exit link harvesting
  1292. }
  1293. bool m_has_link = false;
  1294. };
  1295. struct MakeLinkVector : public LinkMapFunction {
  1296. MakeLinkVector(std::vector<ObjKey>& result)
  1297. : m_links(result)
  1298. {
  1299. }
  1300. bool consume(ObjKey key) override
  1301. {
  1302. m_links.push_back(key);
  1303. return true; // continue evaluation
  1304. }
  1305. std::vector<ObjKey>& m_links;
  1306. };
  1307. struct UnaryLinkResult : public LinkMapFunction {
  1308. bool consume(ObjKey key) override
  1309. {
  1310. m_result = key;
  1311. return false; // exit search, only one result ever expected
  1312. }
  1313. ObjKey m_result;
  1314. };
  1315. struct CountLinks : public LinkMapFunction {
  1316. bool consume(ObjKey) override
  1317. {
  1318. m_link_count++;
  1319. return true;
  1320. }
  1321. size_t result() const
  1322. {
  1323. return m_link_count;
  1324. }
  1325. size_t m_link_count = 0;
  1326. };
  1327. struct CountBacklinks : public LinkMapFunction {
  1328. CountBacklinks(ConstTableRef t)
  1329. : m_table(t)
  1330. {
  1331. }
  1332. bool consume(ObjKey key) override
  1333. {
  1334. m_link_count += m_table.unchecked_ptr()->get_object(key).get_backlink_count();
  1335. return true;
  1336. }
  1337. size_t result() const
  1338. {
  1339. return m_link_count;
  1340. }
  1341. ConstTableRef m_table;
  1342. size_t m_link_count = 0;
  1343. };
  1344. /*
  1345. The LinkMap and LinkMapFunction classes are used for query conditions on links themselves (contrary to conditions on
  1346. the value payload they point at).
  1347. MapLink::map_links() takes a row index of the link array as argument and follows any link chain stated in the query
  1348. (through the link()->link() methods) until the final payload table is reached, and then applies LinkMapFunction on
  1349. the linked-to key(s).
  1350. If all link columns are type_Link, then LinkMapFunction is only invoked for a single key. If one or more
  1351. columns are type_LinkList, then it may result in multiple keys.
  1352. The reason we use this map pattern is that we can exit the link-tree-traversal as early as possible, e.g. when we've
  1353. found the first link that points to key '5'. Other solutions could be a std::vector<ColKey> harvest_all_links(), or an
  1354. iterator pattern. First solution can't exit, second solution requires internal state.
  1355. */
  1356. class LinkMap final {
  1357. public:
  1358. LinkMap() = default;
  1359. LinkMap(ConstTableRef table, std::vector<ColKey> columns)
  1360. : m_link_column_keys(std::move(columns))
  1361. {
  1362. set_base_table(table);
  1363. }
  1364. LinkMap(LinkMap const& other)
  1365. {
  1366. m_link_column_keys = other.m_link_column_keys;
  1367. m_tables = other.m_tables;
  1368. m_link_types = other.m_link_types;
  1369. m_only_unary_links = other.m_only_unary_links;
  1370. }
  1371. size_t get_nb_hops() const
  1372. {
  1373. return m_link_column_keys.size();
  1374. }
  1375. bool has_links() const
  1376. {
  1377. return m_link_column_keys.size() > 0;
  1378. }
  1379. ColKey get_first_column_key() const
  1380. {
  1381. REALM_ASSERT(has_links());
  1382. return m_link_column_keys[0];
  1383. }
  1384. void set_base_table(ConstTableRef table);
  1385. void set_cluster(const Cluster* cluster)
  1386. {
  1387. Allocator& alloc = get_base_table()->get_alloc();
  1388. m_array_ptr = nullptr;
  1389. switch (m_link_types[0]) {
  1390. case col_type_Link:
  1391. if (m_link_column_keys[0].is_dictionary()) {
  1392. m_array_ptr = LeafPtr(new (&m_storage.m_dict) ArrayInteger(alloc));
  1393. }
  1394. else {
  1395. m_array_ptr = LeafPtr(new (&m_storage.m_list) ArrayKey(alloc));
  1396. }
  1397. break;
  1398. case col_type_LinkList:
  1399. m_array_ptr = LeafPtr(new (&m_storage.m_linklist) ArrayList(alloc));
  1400. break;
  1401. case col_type_BackLink:
  1402. m_array_ptr = LeafPtr(new (&m_storage.m_backlink) ArrayBacklink(alloc));
  1403. break;
  1404. default:
  1405. break;
  1406. }
  1407. // m_tables[0]->report_invalid_key(m_link_column_keys[0]);
  1408. cluster->init_leaf(m_link_column_keys[0], m_array_ptr.get());
  1409. m_leaf_ptr = m_array_ptr.get();
  1410. }
  1411. void collect_dependencies(std::vector<TableKey>& tables) const;
  1412. std::string description(util::serializer::SerialisationState& state) const;
  1413. ObjKey get_unary_link_or_not_found(size_t index) const
  1414. {
  1415. REALM_ASSERT(m_only_unary_links);
  1416. UnaryLinkResult res;
  1417. map_links(index, res);
  1418. return res.m_result;
  1419. }
  1420. std::vector<ObjKey> get_links(size_t index) const
  1421. {
  1422. std::vector<ObjKey> res;
  1423. get_links(index, res);
  1424. return res;
  1425. }
  1426. std::vector<ObjKey> get_origin_ndxs(ObjKey key, size_t column = 0) const;
  1427. size_t count_links(size_t row) const
  1428. {
  1429. CountLinks counter;
  1430. map_links(row, counter);
  1431. return counter.result();
  1432. }
  1433. size_t count_all_backlinks(size_t row) const
  1434. {
  1435. CountBacklinks counter(get_target_table());
  1436. map_links(row, counter);
  1437. return counter.result();
  1438. }
  1439. void map_links(size_t row, LinkMapFunction& lm) const
  1440. {
  1441. map_links(0, row, lm);
  1442. }
  1443. bool only_unary_links() const
  1444. {
  1445. return m_only_unary_links;
  1446. }
  1447. ConstTableRef get_base_table() const
  1448. {
  1449. return m_tables.empty() ? nullptr : m_tables[0];
  1450. }
  1451. ConstTableRef get_target_table() const
  1452. {
  1453. REALM_ASSERT(!m_tables.empty());
  1454. return m_tables.back();
  1455. }
  1456. bool links_exist() const
  1457. {
  1458. return !m_link_column_keys.empty();
  1459. }
  1460. static ref_type get_ref(const ArrayPayload* array_payload, ColumnType type, size_t ndx);
  1461. private:
  1462. void map_links(size_t column, ObjKey key, LinkMapFunction& lm) const;
  1463. void map_links(size_t column, size_t row, LinkMapFunction& lm) const;
  1464. void get_links(size_t row, std::vector<ObjKey>& result) const
  1465. {
  1466. MakeLinkVector mlv = MakeLinkVector(result);
  1467. map_links(row, mlv);
  1468. }
  1469. mutable std::vector<ColKey> m_link_column_keys;
  1470. std::vector<ColumnType> m_link_types;
  1471. std::vector<ConstTableRef> m_tables;
  1472. bool m_only_unary_links = true;
  1473. // Leaf cache
  1474. using LeafPtr = std::unique_ptr<ArrayPayload, PlacementDelete>;
  1475. union Storage {
  1476. typename std::aligned_storage<sizeof(ArrayKey), alignof(ArrayKey)>::type m_list;
  1477. typename std::aligned_storage<sizeof(ArrayInteger), alignof(ArrayKey)>::type m_dict;
  1478. typename std::aligned_storage<sizeof(ArrayList), alignof(ArrayList)>::type m_linklist;
  1479. typename std::aligned_storage<sizeof(ArrayList), alignof(ArrayList)>::type m_backlink;
  1480. };
  1481. Storage m_storage;
  1482. LeafPtr m_array_ptr;
  1483. const ArrayPayload* m_leaf_ptr = nullptr;
  1484. template <class>
  1485. friend Query compare(const Subexpr2<Link>&, const Obj&);
  1486. };
  1487. template <class T>
  1488. Value<T> make_value_for_link(bool only_unary_links, size_t size)
  1489. {
  1490. Value<T> value;
  1491. if (only_unary_links) {
  1492. REALM_ASSERT(size <= 1);
  1493. value.init(false, 1);
  1494. value.m_storage.set_null(0);
  1495. }
  1496. else {
  1497. value.init(true, size);
  1498. }
  1499. return value;
  1500. }
  1501. // This class can be used as untyped base for expressions that handle object properties
  1502. class ObjPropertyBase {
  1503. public:
  1504. ObjPropertyBase(ColKey column, ConstTableRef table, std::vector<ColKey> links,
  1505. util::Optional<ExpressionComparisonType> type)
  1506. : m_link_map(table, std::move(links))
  1507. , m_column_key(column)
  1508. , m_comparison_type(type)
  1509. {
  1510. }
  1511. ObjPropertyBase(const ObjPropertyBase& other)
  1512. : m_link_map(other.m_link_map)
  1513. , m_column_key(other.m_column_key)
  1514. , m_comparison_type(other.m_comparison_type)
  1515. {
  1516. }
  1517. ObjPropertyBase(ColKey column, const LinkMap& link_map, util::Optional<ExpressionComparisonType> type)
  1518. : m_link_map(link_map)
  1519. , m_column_key(column)
  1520. , m_comparison_type(type)
  1521. {
  1522. }
  1523. bool links_exist() const
  1524. {
  1525. return m_link_map.has_links();
  1526. }
  1527. bool only_unary_links() const
  1528. {
  1529. return m_link_map.only_unary_links();
  1530. }
  1531. bool is_nullable() const
  1532. {
  1533. return m_column_key.get_attrs().test(col_attr_Nullable);
  1534. }
  1535. const LinkMap& get_link_map() const
  1536. {
  1537. return m_link_map;
  1538. }
  1539. ColKey column_key() const noexcept
  1540. {
  1541. return m_column_key;
  1542. }
  1543. protected:
  1544. LinkMap m_link_map;
  1545. // Column index of payload column of m_table
  1546. mutable ColKey m_column_key;
  1547. util::Optional<ExpressionComparisonType> m_comparison_type; // Any, All, None
  1548. };
  1549. // Combines Subexpr2<T> and ObjPropertyBase
  1550. // Implements virtual functions defined in Expression/Subexpr
  1551. template <class T>
  1552. class ObjPropertyExpr : public Subexpr2<T>, public ObjPropertyBase {
  1553. public:
  1554. using ObjPropertyBase::ObjPropertyBase;
  1555. bool has_multiple_values() const override
  1556. {
  1557. return m_link_map.has_links() && !m_link_map.only_unary_links();
  1558. }
  1559. ConstTableRef get_base_table() const final
  1560. {
  1561. return m_link_map.get_base_table();
  1562. }
  1563. void set_base_table(ConstTableRef table) final
  1564. {
  1565. if (table != get_base_table()) {
  1566. m_link_map.set_base_table(table);
  1567. }
  1568. }
  1569. bool has_search_index() const final
  1570. {
  1571. auto target_table = m_link_map.get_target_table();
  1572. return target_table->search_index_type(m_column_key) == IndexType::General;
  1573. }
  1574. std::vector<ObjKey> find_all(Mixed value) const final
  1575. {
  1576. std::vector<ObjKey> ret;
  1577. std::vector<ObjKey> result;
  1578. if (value.is_null() && !m_column_key.is_nullable()) {
  1579. return ret;
  1580. }
  1581. if (m_link_map.get_target_table()->get_primary_key_column() == m_column_key) {
  1582. // Only one object with a given key would be possible
  1583. if (auto k = m_link_map.get_target_table()->find_primary_key(value))
  1584. result.push_back(k);
  1585. }
  1586. else {
  1587. StringIndex* index = m_link_map.get_target_table()->get_search_index(m_column_key);
  1588. REALM_ASSERT(index);
  1589. if (value.is_null()) {
  1590. index->find_all(result, realm::null{});
  1591. }
  1592. else {
  1593. T val = value.get<T>();
  1594. index->find_all(result, val);
  1595. }
  1596. }
  1597. for (ObjKey k : result) {
  1598. auto ndxs = m_link_map.get_origin_ndxs(k);
  1599. ret.insert(ret.end(), ndxs.begin(), ndxs.end());
  1600. }
  1601. return ret;
  1602. }
  1603. void collect_dependencies(std::vector<TableKey>& tables) const final
  1604. {
  1605. m_link_map.collect_dependencies(tables);
  1606. }
  1607. std::string description(util::serializer::SerialisationState& state) const override
  1608. {
  1609. return state.describe_expression_type(m_comparison_type) + state.describe_columns(m_link_map, m_column_key);
  1610. }
  1611. util::Optional<ExpressionComparisonType> get_comparison_type() const final
  1612. {
  1613. return m_comparison_type;
  1614. }
  1615. std::unique_ptr<Subexpr> clone() const override
  1616. {
  1617. return make_subexpr<Columns<T>>(static_cast<const Columns<T>&>(*this));
  1618. }
  1619. };
  1620. // If we add a new Realm type T and quickly want Query support for it, then simply inherit from it like
  1621. // `template <> class Columns<T> : public SimpleQuerySupport<T>` and you're done. Any operators of the set
  1622. // { ==, >=, <=, !=, >, < } that are supported by T will be supported by the "query expression syntax"
  1623. // automatically. NOTE: This method of Query support will be slow because it goes through Table::get<T>.
  1624. // To get faster Query support, either add SequentialGetter support (faster) or create a query_engine.hpp
  1625. // node for it (super fast).
  1626. template <class T>
  1627. class SimpleQuerySupport : public ObjPropertyExpr<T> {
  1628. public:
  1629. using ObjPropertyExpr<T>::links_exist;
  1630. SimpleQuerySupport(ColKey column, ConstTableRef table, std::vector<ColKey> links = {},
  1631. util::Optional<ExpressionComparisonType> type = util::none)
  1632. : ObjPropertyExpr<T>(column, table, std::move(links), type)
  1633. {
  1634. }
  1635. void set_cluster(const Cluster* cluster) override
  1636. {
  1637. m_array_ptr = nullptr;
  1638. m_leaf_ptr = nullptr;
  1639. if (links_exist()) {
  1640. m_link_map.set_cluster(cluster);
  1641. }
  1642. else {
  1643. // Create new Leaf
  1644. m_array_ptr = LeafPtr(new (&m_leaf_cache_storage) LeafType(m_link_map.get_base_table()->get_alloc()));
  1645. cluster->init_leaf(m_column_key, m_array_ptr.get());
  1646. m_leaf_ptr = m_array_ptr.get();
  1647. }
  1648. }
  1649. void evaluate(size_t index, ValueBase& destination) override
  1650. {
  1651. if (links_exist()) {
  1652. REALM_ASSERT(m_leaf_ptr == nullptr);
  1653. if (m_link_map.only_unary_links()) {
  1654. REALM_ASSERT(destination.size() == 1);
  1655. REALM_ASSERT(!destination.m_from_list);
  1656. destination.set_null(0);
  1657. auto link_translation_key = this->m_link_map.get_unary_link_or_not_found(index);
  1658. if (link_translation_key) {
  1659. const Obj obj = m_link_map.get_target_table()->get_object(link_translation_key);
  1660. if constexpr (realm::is_any_v<T, ObjectId, UUID>) {
  1661. auto opt_val = obj.get<util::Optional<T>>(m_column_key);
  1662. if (opt_val) {
  1663. destination.set(0, *opt_val);
  1664. }
  1665. else {
  1666. destination.set_null(0);
  1667. }
  1668. }
  1669. else {
  1670. destination.set(0, obj.get<T>(m_column_key));
  1671. }
  1672. }
  1673. }
  1674. else {
  1675. std::vector<ObjKey> links = m_link_map.get_links(index);
  1676. destination.init(true, links.size());
  1677. for (size_t t = 0; t < links.size(); t++) {
  1678. const Obj obj = m_link_map.get_target_table()->get_object(links[t]);
  1679. if constexpr (realm::is_any_v<T, ObjectId, UUID>) {
  1680. auto opt_val = obj.get<util::Optional<T>>(m_column_key);
  1681. if (opt_val) {
  1682. destination.set(t, *opt_val);
  1683. }
  1684. else {
  1685. destination.set_null(t);
  1686. }
  1687. }
  1688. else {
  1689. destination.set(t, obj.get<T>(m_column_key));
  1690. }
  1691. }
  1692. }
  1693. }
  1694. else {
  1695. // Not a link column
  1696. REALM_ASSERT(m_leaf_ptr != nullptr);
  1697. REALM_ASSERT(destination.size() == 1);
  1698. REALM_ASSERT(!destination.m_from_list);
  1699. if (m_leaf_ptr->is_null(index)) {
  1700. destination.set_null(0);
  1701. }
  1702. else {
  1703. destination.set(0, m_leaf_ptr->get(index));
  1704. }
  1705. }
  1706. }
  1707. void evaluate(ObjKey key, ValueBase& destination) override
  1708. {
  1709. Value<T>& d = static_cast<Value<T>&>(destination);
  1710. d.set(0, m_link_map.get_target_table()->get_object(key).template get<T>(m_column_key));
  1711. }
  1712. SimpleQuerySupport(const SimpleQuerySupport& other)
  1713. : ObjPropertyExpr<T>(other)
  1714. {
  1715. }
  1716. SizeOperator<T> size()
  1717. {
  1718. return SizeOperator<T>(this->clone());
  1719. }
  1720. TypeOfValueOperator<T> type_of_value()
  1721. {
  1722. return TypeOfValueOperator<T>(this->clone());
  1723. }
  1724. private:
  1725. using ObjPropertyExpr<T>::m_link_map;
  1726. using ObjPropertyExpr<T>::m_column_key;
  1727. // Leaf cache
  1728. using LeafType = typename ColumnTypeTraits<T>::cluster_leaf_type;
  1729. using LeafCacheStorage = typename std::aligned_storage<sizeof(LeafType), alignof(LeafType)>::type;
  1730. using LeafPtr = std::unique_ptr<LeafType, PlacementDelete>;
  1731. LeafCacheStorage m_leaf_cache_storage;
  1732. LeafPtr m_array_ptr;
  1733. LeafType* m_leaf_ptr = nullptr;
  1734. };
  1735. template <>
  1736. class Columns<Timestamp> : public SimpleQuerySupport<Timestamp> {
  1737. using SimpleQuerySupport::SimpleQuerySupport;
  1738. };
  1739. template <>
  1740. class Columns<BinaryData> : public SimpleQuerySupport<BinaryData> {
  1741. using SimpleQuerySupport::SimpleQuerySupport;
  1742. friend Query operator==(const Columns<BinaryData>& left, BinaryData right)
  1743. {
  1744. return create<Equal>(right, left);
  1745. }
  1746. friend Query operator==(BinaryData left, const Columns<BinaryData>& right)
  1747. {
  1748. return create<Equal>(left, right);
  1749. }
  1750. friend Query operator!=(const Columns<BinaryData>& left, BinaryData right)
  1751. {
  1752. return create<NotEqual>(right, left);
  1753. }
  1754. friend Query operator!=(BinaryData left, const Columns<BinaryData>& right)
  1755. {
  1756. return create<NotEqual>(left, right);
  1757. }
  1758. friend Query operator==(const Columns<BinaryData>& left, realm::null)
  1759. {
  1760. return create<Equal>(BinaryData(), left);
  1761. }
  1762. friend Query operator==(realm::null, const Columns<BinaryData>& right)
  1763. {
  1764. return create<Equal>(BinaryData(), right);
  1765. }
  1766. friend Query operator!=(const Columns<BinaryData>& left, realm::null)
  1767. {
  1768. return create<NotEqual>(BinaryData(), left);
  1769. }
  1770. friend Query operator!=(realm::null, const Columns<BinaryData>& right)
  1771. {
  1772. return create<NotEqual>(BinaryData(), right);
  1773. }
  1774. };
  1775. template <>
  1776. class Columns<ObjectId> : public SimpleQuerySupport<ObjectId> {
  1777. using SimpleQuerySupport::SimpleQuerySupport;
  1778. };
  1779. template <>
  1780. class Columns<Decimal128> : public SimpleQuerySupport<Decimal128> {
  1781. using SimpleQuerySupport::SimpleQuerySupport;
  1782. };
  1783. template <>
  1784. class Columns<Mixed> : public SimpleQuerySupport<Mixed> {
  1785. using SimpleQuerySupport::SimpleQuerySupport;
  1786. };
  1787. template <>
  1788. class Columns<UUID> : public SimpleQuerySupport<UUID> {
  1789. using SimpleQuerySupport::SimpleQuerySupport;
  1790. };
  1791. template <class T, class S, class I>
  1792. inline std::enable_if_t<!realm::is_any_v<T, StringData, realm::null, const char*, std::string>, Query>
  1793. string_compare(const Subexpr2<StringData>& left, T right, bool)
  1794. {
  1795. return make_expression<Compare<Equal>>(right.clone(), left.clone());
  1796. }
  1797. template <class T, class S, class I>
  1798. inline std::enable_if_t<realm::is_any_v<T, StringData, realm::null, const char*, std::string>, Query>
  1799. string_compare(const Subexpr2<StringData>& left, T right, bool case_sensitive)
  1800. {
  1801. StringData sd(right);
  1802. if (case_sensitive)
  1803. return create<S>(sd, left);
  1804. else
  1805. return create<I>(sd, left);
  1806. }
  1807. template <class S, class I>
  1808. Query string_compare(const Subexpr2<StringData>& left, const Subexpr2<StringData>& right, bool case_sensitive)
  1809. {
  1810. if (case_sensitive)
  1811. return make_expression<Compare<S>>(right.clone(), left.clone());
  1812. else
  1813. return make_expression<Compare<I>>(right.clone(), left.clone());
  1814. }
  1815. template <>
  1816. class Columns<StringData> : public SimpleQuerySupport<StringData> {
  1817. public:
  1818. Columns(ColKey column, ConstTableRef table, std::vector<ColKey> links = {},
  1819. util::Optional<ExpressionComparisonType> type = util::none)
  1820. : SimpleQuerySupport(column, table, links, type)
  1821. {
  1822. }
  1823. Columns(Columns const& other)
  1824. : SimpleQuerySupport(other)
  1825. {
  1826. }
  1827. Columns(Columns&& other) noexcept
  1828. : SimpleQuerySupport(other)
  1829. {
  1830. }
  1831. Query fulltext(StringData sd) const;
  1832. using SimpleQuerySupport::size;
  1833. // Columns<String> == Columns<String>
  1834. friend Query operator==(const Columns<StringData>& left, const Columns<StringData>& right)
  1835. {
  1836. return string_compare<Equal, EqualIns>(left, right, true);
  1837. }
  1838. // Columns<String> != Columns<String>
  1839. friend Query operator!=(const Columns<StringData>& left, const Columns<StringData>& right)
  1840. {
  1841. return string_compare<NotEqual, NotEqualIns>(left, right, true);
  1842. }
  1843. // String == Columns<String>
  1844. template <class T>
  1845. friend Query operator==(T left, const Columns<StringData>& right)
  1846. {
  1847. return operator==(right, left);
  1848. }
  1849. // String != Columns<String>
  1850. template <class T>
  1851. friend Query operator!=(T left, const Columns<StringData>& right)
  1852. {
  1853. return operator!=(right, left);
  1854. }
  1855. // Columns<String> == String
  1856. template <class T>
  1857. friend Query operator==(const Columns<StringData>& left, T right)
  1858. {
  1859. return string_compare<T, Equal, EqualIns>(left, right, true);
  1860. }
  1861. // Columns<String> != String
  1862. template <class T>
  1863. friend Query operator!=(const Columns<StringData>& left, T right)
  1864. {
  1865. return string_compare<T, NotEqual, NotEqualIns>(left, right, true);
  1866. }
  1867. };
  1868. template <class T, class S, class I>
  1869. Query binary_compare(const Subexpr2<BinaryData>& left, T right, bool case_sensitive)
  1870. {
  1871. BinaryData data(right);
  1872. if (case_sensitive)
  1873. return create<S>(data, left);
  1874. else
  1875. return create<I>(data, left);
  1876. }
  1877. template <class S, class I>
  1878. Query binary_compare(const Subexpr2<BinaryData>& left, const Subexpr2<BinaryData>& right, bool case_sensitive)
  1879. {
  1880. if (case_sensitive)
  1881. return make_expression<Compare<S>>(right.clone(), left.clone());
  1882. else
  1883. return make_expression<Compare<I>>(right.clone(), left.clone());
  1884. }
  1885. template <class T, class S, class I>
  1886. Query mixed_compare(const Subexpr2<Mixed>& left, T right, bool case_sensitive)
  1887. {
  1888. Mixed data(right);
  1889. if (case_sensitive)
  1890. return create<S>(data, left);
  1891. else
  1892. return create<I>(data, left);
  1893. }
  1894. template <class S, class I>
  1895. Query mixed_compare(const Subexpr2<Mixed>& left, const Subexpr2<Mixed>& right, bool case_sensitive)
  1896. {
  1897. if (case_sensitive)
  1898. return make_expression<Compare<S>>(right.clone(), left.clone());
  1899. else
  1900. return make_expression<Compare<I>>(right.clone(), left.clone());
  1901. }
  1902. // This class is intended to perform queries on the *pointers* of links, contrary to performing queries on *payload*
  1903. // in linked-to tables. Queries can be "find first link that points at row X" or "find first null-link". Currently
  1904. // only "find first null link" and "find first non-null link" is supported. More will be added later. When we add
  1905. // more, I propose to remove the <bool has_links> template argument from this class and instead template it by
  1906. // a criteria-class (like the FindNullLinks class below in find_first()) in some generalized fashion.
  1907. template <bool has_links>
  1908. class UnaryLinkCompare : public Expression {
  1909. public:
  1910. UnaryLinkCompare(const LinkMap& lm)
  1911. : m_link_map(lm)
  1912. {
  1913. }
  1914. void set_base_table(ConstTableRef table) override
  1915. {
  1916. m_link_map.set_base_table(table);
  1917. }
  1918. void set_cluster(const Cluster* cluster) override
  1919. {
  1920. m_link_map.set_cluster(cluster);
  1921. }
  1922. void collect_dependencies(std::vector<TableKey>& tables) const override
  1923. {
  1924. m_link_map.collect_dependencies(tables);
  1925. }
  1926. // Return main table of query (table on which table->where()... is invoked). Note that this is not the same as
  1927. // any linked-to payload tables
  1928. ConstTableRef get_base_table() const override
  1929. {
  1930. return m_link_map.get_base_table();
  1931. }
  1932. size_t find_first(size_t start, size_t end) const override
  1933. {
  1934. for (; start < end;) {
  1935. FindNullLinks fnl;
  1936. m_link_map.map_links(start, fnl);
  1937. if (fnl.m_has_link == has_links)
  1938. return start;
  1939. start++;
  1940. }
  1941. return not_found;
  1942. }
  1943. virtual std::string description(util::serializer::SerialisationState& state) const override
  1944. {
  1945. return state.describe_columns(m_link_map, ColKey()) + (has_links ? " != NULL" : " == NULL");
  1946. }
  1947. std::unique_ptr<Expression> clone() const override
  1948. {
  1949. return std::unique_ptr<Expression>(new UnaryLinkCompare(*this));
  1950. }
  1951. private:
  1952. UnaryLinkCompare(const UnaryLinkCompare& other)
  1953. : Expression(other)
  1954. , m_link_map(other.m_link_map)
  1955. {
  1956. }
  1957. mutable LinkMap m_link_map;
  1958. };
  1959. class LinkCount : public Subexpr2<Int> {
  1960. public:
  1961. LinkCount(const LinkMap& link_map)
  1962. : m_link_map(link_map)
  1963. {
  1964. }
  1965. LinkCount(LinkCount const& other)
  1966. : Subexpr2<Int>(other)
  1967. , m_link_map(other.m_link_map)
  1968. {
  1969. }
  1970. std::unique_ptr<Subexpr> clone() const override
  1971. {
  1972. return make_subexpr<LinkCount>(*this);
  1973. }
  1974. ConstTableRef get_base_table() const override
  1975. {
  1976. return m_link_map.get_base_table();
  1977. }
  1978. void set_base_table(ConstTableRef table) override
  1979. {
  1980. m_link_map.set_base_table(table);
  1981. }
  1982. void set_cluster(const Cluster* cluster) override
  1983. {
  1984. m_link_map.set_cluster(cluster);
  1985. }
  1986. void collect_dependencies(std::vector<TableKey>& tables) const override
  1987. {
  1988. m_link_map.collect_dependencies(tables);
  1989. }
  1990. void evaluate(size_t index, ValueBase& destination) override
  1991. {
  1992. size_t count = m_link_map.count_links(index);
  1993. destination = Value<int64_t>(count);
  1994. }
  1995. virtual std::string description(util::serializer::SerialisationState& state) const override
  1996. {
  1997. return state.describe_columns(m_link_map, ColKey()) + util::serializer::value_separator + "@count";
  1998. }
  1999. private:
  2000. LinkMap m_link_map;
  2001. };
  2002. // Gives a count of all backlinks across all columns for the specified row.
  2003. // The unused template parameter is a hack to avoid a circular dependency between table.hpp and query_expression.hpp.
  2004. template <class>
  2005. class BacklinkCount : public Subexpr2<Int> {
  2006. public:
  2007. BacklinkCount(const LinkMap& link_map)
  2008. : m_link_map(link_map)
  2009. {
  2010. }
  2011. BacklinkCount(LinkMap&& link_map)
  2012. : m_link_map(std::move(link_map))
  2013. {
  2014. }
  2015. BacklinkCount(ConstTableRef table, std::vector<ColKey> links = {})
  2016. : m_link_map(table, std::move(links))
  2017. {
  2018. }
  2019. BacklinkCount(BacklinkCount const& other)
  2020. : Subexpr2<Int>(other)
  2021. , m_link_map(other.m_link_map)
  2022. {
  2023. }
  2024. std::unique_ptr<Subexpr> clone() const override
  2025. {
  2026. return make_subexpr<BacklinkCount<Int>>(*this);
  2027. }
  2028. ConstTableRef get_base_table() const override
  2029. {
  2030. return m_link_map.get_base_table();
  2031. }
  2032. void set_base_table(ConstTableRef table) override
  2033. {
  2034. m_link_map.set_base_table(table);
  2035. }
  2036. void set_cluster(const Cluster* cluster) override
  2037. {
  2038. if (m_link_map.has_links()) {
  2039. m_link_map.set_cluster(cluster);
  2040. }
  2041. else {
  2042. m_keys = cluster->get_key_array();
  2043. m_offset = cluster->get_offset();
  2044. }
  2045. }
  2046. void collect_dependencies(std::vector<TableKey>& tables) const override
  2047. {
  2048. m_link_map.collect_dependencies(tables);
  2049. }
  2050. void evaluate(size_t index, ValueBase& destination) override
  2051. {
  2052. size_t count;
  2053. if (m_link_map.has_links()) {
  2054. count = m_link_map.count_all_backlinks(index);
  2055. }
  2056. else {
  2057. ObjKey key(m_keys->get(index) + m_offset);
  2058. const Obj obj = m_link_map.get_base_table()->get_object(key);
  2059. count = obj.get_backlink_count();
  2060. }
  2061. destination = Value<int64_t>(count);
  2062. }
  2063. virtual std::string description(util::serializer::SerialisationState& state) const override
  2064. {
  2065. std::string s;
  2066. if (m_link_map.links_exist()) {
  2067. s += state.describe_columns(m_link_map, ColKey()) + util::serializer::value_separator;
  2068. }
  2069. s += "@links.@count";
  2070. return s;
  2071. }
  2072. private:
  2073. const ClusterKeyArray* m_keys = nullptr;
  2074. uint64_t m_offset = 0;
  2075. LinkMap m_link_map;
  2076. };
  2077. template <class T, class TExpr>
  2078. class SizeOperator : public Subexpr2<Int> {
  2079. public:
  2080. SizeOperator(std::unique_ptr<TExpr> left)
  2081. : m_expr(std::move(left))
  2082. {
  2083. }
  2084. SizeOperator(const SizeOperator& other)
  2085. : m_expr(other.m_expr->clone())
  2086. {
  2087. }
  2088. // See comment in base class
  2089. void set_base_table(ConstTableRef table) override
  2090. {
  2091. m_expr->set_base_table(table);
  2092. }
  2093. void set_cluster(const Cluster* cluster) override
  2094. {
  2095. m_expr->set_cluster(cluster);
  2096. }
  2097. // Recursively fetch tables of columns in expression tree. Used when user first builds a stand-alone expression
  2098. // and binds it to a Query at a later time
  2099. ConstTableRef get_base_table() const override
  2100. {
  2101. return m_expr->get_base_table();
  2102. }
  2103. // destination = operator(left)
  2104. void evaluate(size_t index, ValueBase& destination) override
  2105. {
  2106. Value<T> v;
  2107. m_expr->evaluate(index, v);
  2108. size_t sz = v.size();
  2109. destination.init(v.m_from_list, sz);
  2110. for (size_t i = 0; i < sz; i++) {
  2111. auto elem = v[i].template get<T>();
  2112. if constexpr (std::is_same_v<T, int64_t>) {
  2113. // This is the size of a list
  2114. destination.set(i, elem);
  2115. }
  2116. else {
  2117. if (!elem) {
  2118. destination.set_null(i);
  2119. }
  2120. else {
  2121. destination.set(i, int64_t(elem.size()));
  2122. }
  2123. }
  2124. }
  2125. }
  2126. std::string description(util::serializer::SerialisationState& state) const override
  2127. {
  2128. if (m_expr) {
  2129. return m_expr->description(state) + util::serializer::value_separator + "@size";
  2130. }
  2131. return "@size";
  2132. }
  2133. std::unique_ptr<Subexpr> clone() const override
  2134. {
  2135. return std::unique_ptr<Subexpr>(new SizeOperator(*this));
  2136. }
  2137. private:
  2138. std::unique_ptr<TExpr> m_expr;
  2139. };
  2140. template <class T>
  2141. class TypeOfValueOperator : public Subexpr2<TypeOfValue> {
  2142. public:
  2143. TypeOfValueOperator(std::unique_ptr<Subexpr> left)
  2144. : m_expr(std::move(left))
  2145. {
  2146. }
  2147. TypeOfValueOperator(const TypeOfValueOperator& other)
  2148. : m_expr(other.m_expr->clone())
  2149. {
  2150. }
  2151. util::Optional<ExpressionComparisonType> get_comparison_type() const override
  2152. {
  2153. return m_expr->get_comparison_type();
  2154. }
  2155. // See comment in base class
  2156. void set_base_table(ConstTableRef table) override
  2157. {
  2158. m_expr->set_base_table(table);
  2159. }
  2160. void set_cluster(const Cluster* cluster) override
  2161. {
  2162. m_expr->set_cluster(cluster);
  2163. }
  2164. // Recursively fetch tables of columns in expression tree. Used when user first builds a stand-alone expression
  2165. // and binds it to a Query at a later time
  2166. ConstTableRef get_base_table() const override
  2167. {
  2168. return m_expr->get_base_table();
  2169. }
  2170. // destination = operator(left)
  2171. void evaluate(size_t index, ValueBase& destination) override
  2172. {
  2173. Value<T> v;
  2174. m_expr->evaluate(index, v);
  2175. size_t sz = v.size();
  2176. destination.init(v.m_from_list, sz);
  2177. for (size_t i = 0; i < sz; i++) {
  2178. auto elem = v[i].template get<T>();
  2179. destination.set(i, TypeOfValue(elem));
  2180. }
  2181. }
  2182. std::string description(util::serializer::SerialisationState& state) const override
  2183. {
  2184. if (m_expr) {
  2185. return m_expr->description(state) + util::serializer::value_separator + "@type";
  2186. }
  2187. return "@type";
  2188. }
  2189. std::unique_ptr<Subexpr> clone() const override
  2190. {
  2191. return std::unique_ptr<Subexpr>(new TypeOfValueOperator(*this));
  2192. }
  2193. private:
  2194. std::unique_ptr<Subexpr> m_expr;
  2195. };
  2196. class KeyValue : public Subexpr2<Link> {
  2197. public:
  2198. KeyValue(ObjKey key)
  2199. : m_key(key)
  2200. {
  2201. }
  2202. void set_base_table(ConstTableRef) override {}
  2203. ConstTableRef get_base_table() const override
  2204. {
  2205. return nullptr;
  2206. }
  2207. void evaluate(size_t, ValueBase& destination) override
  2208. {
  2209. destination = Value<ObjKey>(m_key);
  2210. }
  2211. virtual std::string description(util::serializer::SerialisationState&) const override
  2212. {
  2213. return util::serializer::print_value(m_key);
  2214. }
  2215. std::unique_ptr<Subexpr> clone() const override
  2216. {
  2217. return std::unique_ptr<Subexpr>(new KeyValue(*this));
  2218. }
  2219. private:
  2220. KeyValue(const KeyValue& source)
  2221. : m_key(source.m_key)
  2222. {
  2223. }
  2224. ObjKey m_key;
  2225. };
  2226. template <typename T>
  2227. class SubColumns;
  2228. // This is for LinkList and BackLink too since they're declared as typedefs of Link.
  2229. template <>
  2230. class Columns<Link> : public Subexpr2<Link> {
  2231. public:
  2232. Columns(const Columns& other)
  2233. : Subexpr2<Link>(other)
  2234. , m_link_map(other.m_link_map)
  2235. , m_comparison_type(other.m_comparison_type)
  2236. , m_is_list(other.m_is_list)
  2237. {
  2238. }
  2239. Columns(ColKey column_key, ConstTableRef table, const std::vector<ColKey>& links = {},
  2240. util::Optional<ExpressionComparisonType> type = util::none)
  2241. : m_link_map(table, links)
  2242. , m_comparison_type(type)
  2243. , m_is_list(column_key.is_list())
  2244. {
  2245. }
  2246. Query is_null()
  2247. {
  2248. if (m_link_map.get_nb_hops() > 1)
  2249. throw Exception(ErrorCodes::InvalidQuery, "Combining link() and is_null() is currently not supported");
  2250. // Todo, it may be useful to support the above, but we would need to figure out an intuitive behaviour
  2251. return make_expression<UnaryLinkCompare<false>>(m_link_map);
  2252. }
  2253. Query is_not_null()
  2254. {
  2255. if (m_link_map.get_nb_hops() > 1)
  2256. throw Exception(ErrorCodes::InvalidQuery,
  2257. "Combining link() and is_not_null() is currently not supported");
  2258. // Todo, it may be useful to support the above, but we would need to figure out an intuitive behaviour
  2259. return make_expression<UnaryLinkCompare<true>>(m_link_map);
  2260. }
  2261. LinkCount count() const
  2262. {
  2263. return LinkCount(m_link_map);
  2264. }
  2265. template <class T>
  2266. BacklinkCount<T> backlink_count() const
  2267. {
  2268. return BacklinkCount<T>(m_link_map);
  2269. }
  2270. template <typename C>
  2271. SubColumns<C> column(ColKey column_key) const
  2272. {
  2273. // no need to pass along m_comparison_type because the only operations supported from
  2274. // the subsequent SubColumns are aggregate operations such as sum, min, max, avg where
  2275. // having
  2276. REALM_ASSERT_DEBUG(!m_comparison_type);
  2277. return SubColumns<C>(Columns<C>(column_key, m_link_map.get_target_table()), m_link_map);
  2278. }
  2279. const LinkMap& link_map() const
  2280. {
  2281. return m_link_map;
  2282. }
  2283. DataType get_type() const override
  2284. {
  2285. return type_Link;
  2286. }
  2287. ConstTableRef get_target_table() const override
  2288. {
  2289. return link_map().get_target_table();
  2290. }
  2291. bool has_multiple_values() const override
  2292. {
  2293. return m_is_list || !m_link_map.only_unary_links();
  2294. }
  2295. ConstTableRef get_base_table() const override
  2296. {
  2297. return m_link_map.get_base_table();
  2298. }
  2299. void set_base_table(ConstTableRef table) override
  2300. {
  2301. m_link_map.set_base_table(table);
  2302. }
  2303. void set_cluster(const Cluster* cluster) override
  2304. {
  2305. REALM_ASSERT(m_link_map.has_links());
  2306. m_link_map.set_cluster(cluster);
  2307. }
  2308. void collect_dependencies(std::vector<TableKey>& tables) const override
  2309. {
  2310. m_link_map.collect_dependencies(tables);
  2311. }
  2312. std::string description(util::serializer::SerialisationState& state) const override
  2313. {
  2314. return state.describe_expression_type(m_comparison_type) + state.describe_columns(m_link_map, ColKey());
  2315. }
  2316. util::Optional<ExpressionComparisonType> get_comparison_type() const override
  2317. {
  2318. return m_comparison_type;
  2319. }
  2320. std::unique_ptr<Subexpr> clone() const override
  2321. {
  2322. return std::unique_ptr<Subexpr>(new Columns<Link>(*this));
  2323. }
  2324. void evaluate(size_t index, ValueBase& destination) override;
  2325. private:
  2326. LinkMap m_link_map;
  2327. util::Optional<ExpressionComparisonType> m_comparison_type;
  2328. bool m_is_list;
  2329. friend class Table;
  2330. friend class LinkChain;
  2331. };
  2332. template <typename T>
  2333. class ListColumns;
  2334. template <typename T, typename Operation>
  2335. class CollectionColumnAggregate;
  2336. namespace aggregate_operations {
  2337. template <typename T>
  2338. class Minimum;
  2339. template <typename T>
  2340. class Maximum;
  2341. template <typename T>
  2342. class Sum;
  2343. template <typename T>
  2344. class Average;
  2345. } // namespace aggregate_operations
  2346. class ColumnListBase {
  2347. public:
  2348. ColumnListBase(ColKey column_key, ConstTableRef table, const std::vector<ColKey>& links,
  2349. util::Optional<ExpressionComparisonType> type = util::none)
  2350. : m_column_key(column_key)
  2351. , m_link_map(table, links)
  2352. , m_comparison_type(type)
  2353. {
  2354. }
  2355. ColumnListBase(const ColumnListBase& other)
  2356. : m_column_key(other.m_column_key)
  2357. , m_link_map(other.m_link_map)
  2358. , m_comparison_type(other.m_comparison_type)
  2359. {
  2360. }
  2361. void set_cluster(const Cluster* cluster);
  2362. void get_lists(size_t index, Value<int64_t>& destination, size_t nb_elements);
  2363. std::string description(util::serializer::SerialisationState& state) const
  2364. {
  2365. return state.describe_expression_type(m_comparison_type) + state.describe_columns(m_link_map, m_column_key);
  2366. }
  2367. bool links_exist() const
  2368. {
  2369. return m_link_map.has_links();
  2370. }
  2371. virtual SizeOperator<int64_t> size() = 0;
  2372. virtual std::unique_ptr<Subexpr> get_element_length() = 0;
  2373. virtual std::unique_ptr<Subexpr> max_of() = 0;
  2374. virtual std::unique_ptr<Subexpr> min_of() = 0;
  2375. virtual std::unique_ptr<Subexpr> sum_of() = 0;
  2376. virtual std::unique_ptr<Subexpr> avg_of() = 0;
  2377. mutable ColKey m_column_key;
  2378. LinkMap m_link_map;
  2379. // Leaf cache
  2380. using LeafCacheStorage = typename std::aligned_storage<sizeof(ArrayInteger), alignof(Array)>::type;
  2381. using LeafPtr = std::unique_ptr<ArrayInteger, PlacementDelete>;
  2382. LeafCacheStorage m_leaf_cache_storage;
  2383. LeafPtr m_array_ptr;
  2384. ArrayInteger* m_leaf_ptr = nullptr;
  2385. util::Optional<ExpressionComparisonType> m_comparison_type;
  2386. };
  2387. template <typename>
  2388. class ColumnListSize;
  2389. template <typename T>
  2390. class ColumnListElementLength;
  2391. template <typename T>
  2392. class ColumnsCollection : public Subexpr2<T>, public ColumnListBase {
  2393. public:
  2394. ColumnsCollection(ColKey column_key, ConstTableRef table, const std::vector<ColKey>& links = {},
  2395. util::Optional<ExpressionComparisonType> type = util::none)
  2396. : ColumnListBase(column_key, table, links, type)
  2397. , m_is_nullable_storage(this->m_column_key.get_attrs().test(col_attr_Nullable))
  2398. {
  2399. }
  2400. ColumnsCollection(const ColumnsCollection& other)
  2401. : Subexpr2<T>(other)
  2402. , ColumnListBase(other)
  2403. , m_is_nullable_storage(this->m_column_key.get_attrs().test(col_attr_Nullable))
  2404. {
  2405. }
  2406. bool has_multiple_values() const override
  2407. {
  2408. return true;
  2409. }
  2410. ConstTableRef get_base_table() const final
  2411. {
  2412. return m_link_map.get_base_table();
  2413. }
  2414. ConstTableRef get_target_table() const override
  2415. {
  2416. return m_link_map.get_target_table()->get_opposite_table(m_column_key);
  2417. }
  2418. Allocator& get_alloc() const
  2419. {
  2420. return m_link_map.get_target_table()->get_alloc();
  2421. }
  2422. void set_base_table(ConstTableRef table) final
  2423. {
  2424. m_link_map.set_base_table(table);
  2425. }
  2426. void set_cluster(const Cluster* cluster) final
  2427. {
  2428. ColumnListBase::set_cluster(cluster);
  2429. }
  2430. void collect_dependencies(std::vector<TableKey>& tables) const final
  2431. {
  2432. m_link_map.collect_dependencies(tables);
  2433. }
  2434. void evaluate(size_t index, ValueBase& destination) override
  2435. {
  2436. if constexpr (realm::is_any_v<T, ObjectId, Int, Bool, UUID>) {
  2437. if (m_is_nullable_storage) {
  2438. evaluate<util::Optional<T>>(index, destination);
  2439. return;
  2440. }
  2441. }
  2442. evaluate<T>(index, destination);
  2443. }
  2444. std::string description(util::serializer::SerialisationState& state) const override
  2445. {
  2446. return ColumnListBase::description(state);
  2447. }
  2448. util::Optional<ExpressionComparisonType> get_comparison_type() const final
  2449. {
  2450. return ColumnListBase::m_comparison_type;
  2451. }
  2452. SizeOperator<int64_t> size() override;
  2453. ColumnListElementLength<T> element_lengths() const
  2454. {
  2455. return {*this};
  2456. }
  2457. TypeOfValueOperator<T> type_of_value()
  2458. {
  2459. return TypeOfValueOperator<T>(this->clone());
  2460. }
  2461. CollectionColumnAggregate<T, aggregate_operations::Minimum<T>> min() const
  2462. {
  2463. return {*this};
  2464. }
  2465. CollectionColumnAggregate<T, aggregate_operations::Maximum<T>> max() const
  2466. {
  2467. return {*this};
  2468. }
  2469. CollectionColumnAggregate<T, aggregate_operations::Sum<T>> sum() const
  2470. {
  2471. return {*this};
  2472. }
  2473. CollectionColumnAggregate<T, aggregate_operations::Average<T>> average() const
  2474. {
  2475. return {*this};
  2476. }
  2477. std::unique_ptr<Subexpr> max_of() override
  2478. {
  2479. if constexpr (realm::is_any_v<T, Int, Float, Double, Decimal128, Mixed>) {
  2480. return max().clone();
  2481. }
  2482. else {
  2483. return {};
  2484. }
  2485. }
  2486. std::unique_ptr<Subexpr> min_of() override
  2487. {
  2488. if constexpr (realm::is_any_v<T, Int, Float, Double, Decimal128, Mixed>) {
  2489. return min().clone();
  2490. }
  2491. else {
  2492. return {};
  2493. }
  2494. }
  2495. std::unique_ptr<Subexpr> sum_of() override
  2496. {
  2497. if constexpr (realm::is_any_v<T, Int, Float, Double, Decimal128, Mixed>) {
  2498. return sum().clone();
  2499. }
  2500. else {
  2501. return {};
  2502. }
  2503. }
  2504. std::unique_ptr<Subexpr> avg_of() override
  2505. {
  2506. if constexpr (realm::is_any_v<T, Int, Float, Double, Decimal128, Mixed>) {
  2507. return average().clone();
  2508. }
  2509. else {
  2510. return {};
  2511. }
  2512. }
  2513. std::unique_ptr<Subexpr> get_element_length() override
  2514. {
  2515. if constexpr (realm::is_any_v<T, StringData, BinaryData, Mixed>) {
  2516. return element_lengths().clone();
  2517. }
  2518. else {
  2519. return {};
  2520. }
  2521. }
  2522. std::unique_ptr<Subexpr> clone() const override
  2523. {
  2524. return std::unique_ptr<Subexpr>(new ColumnsCollection(*this));
  2525. }
  2526. const bool m_is_nullable_storage;
  2527. private:
  2528. template <typename StorageType>
  2529. void evaluate(size_t index, ValueBase& destination)
  2530. {
  2531. Allocator& alloc = get_alloc();
  2532. Value<int64_t> list_refs;
  2533. get_lists(index, list_refs, 1);
  2534. const bool is_from_list = true;
  2535. std::vector<StorageType> values;
  2536. for (auto&& i : list_refs) {
  2537. ref_type list_ref = to_ref(i.get_int());
  2538. if (list_ref) {
  2539. BPlusTree<StorageType> list(alloc);
  2540. list.init_from_ref(list_ref);
  2541. size_t s = list.size();
  2542. for (size_t j = 0; j < s; j++) {
  2543. values.push_back(list.get(j));
  2544. }
  2545. }
  2546. }
  2547. destination.init(is_from_list, values.size());
  2548. destination.set(values.begin(), values.end());
  2549. }
  2550. };
  2551. template <typename T>
  2552. class Columns<Lst<T>> : public ColumnsCollection<T> {
  2553. public:
  2554. using ColumnsCollection<T>::ColumnsCollection;
  2555. std::unique_ptr<Subexpr> clone() const override
  2556. {
  2557. return make_subexpr<Columns<Lst<T>>>(*this);
  2558. }
  2559. friend class Table;
  2560. friend class LinkChain;
  2561. };
  2562. template <typename T>
  2563. class Columns<Set<T>> : public ColumnsCollection<T> {
  2564. public:
  2565. using ColumnsCollection<T>::ColumnsCollection;
  2566. std::unique_ptr<Subexpr> clone() const override
  2567. {
  2568. return make_subexpr<Columns<Set<T>>>(*this);
  2569. }
  2570. };
  2571. template <>
  2572. class Columns<LnkLst> : public Columns<Lst<ObjKey>> {
  2573. public:
  2574. using Columns<Lst<ObjKey>>::Columns;
  2575. std::unique_ptr<Subexpr> clone() const override
  2576. {
  2577. return make_subexpr<Columns<LnkLst>>(*this);
  2578. }
  2579. };
  2580. template <>
  2581. class Columns<LnkSet> : public Columns<Set<ObjKey>> {
  2582. public:
  2583. using Columns<Set<ObjKey>>::Columns;
  2584. std::unique_ptr<Subexpr> clone() const override
  2585. {
  2586. return make_subexpr<Columns<LnkSet>>(*this);
  2587. }
  2588. };
  2589. // Returns the keys
  2590. class ColumnDictionaryKeys;
  2591. // Returns the values of a given key
  2592. class ColumnDictionaryKey;
  2593. // Returns the values
  2594. template <>
  2595. class Columns<Dictionary> : public ColumnsCollection<Mixed> {
  2596. public:
  2597. Columns(ColKey column, ConstTableRef table, std::vector<ColKey> links = {},
  2598. util::Optional<ExpressionComparisonType> type = util::none)
  2599. : ColumnsCollection<Mixed>(column, table, std::move(links), type)
  2600. {
  2601. m_key_type = m_link_map.get_target_table()->get_dictionary_key_type(column);
  2602. }
  2603. DataType get_key_type() const
  2604. {
  2605. return m_key_type;
  2606. }
  2607. ColumnDictionaryKey key(const Mixed& key_value);
  2608. ColumnDictionaryKeys keys();
  2609. SizeOperator<int64_t> size() override;
  2610. std::unique_ptr<Subexpr> get_element_length() override
  2611. {
  2612. // Not supported for Dictionary
  2613. return {};
  2614. }
  2615. void evaluate(size_t index, ValueBase& destination) override;
  2616. std::unique_ptr<Subexpr> clone() const override
  2617. {
  2618. return make_subexpr<Columns<Dictionary>>(*this);
  2619. }
  2620. Columns(Columns const& other)
  2621. : ColumnsCollection<Mixed>(other)
  2622. , m_key_type(other.m_key_type)
  2623. {
  2624. }
  2625. protected:
  2626. DataType m_key_type;
  2627. };
  2628. class ColumnDictionaryKey : public Columns<Dictionary> {
  2629. public:
  2630. ColumnDictionaryKey(Mixed key_value, const Columns<Dictionary>& dict)
  2631. : Columns<Dictionary>(dict)
  2632. {
  2633. init_key(key_value);
  2634. }
  2635. ColumnDictionaryKey& property(const std::string& prop)
  2636. {
  2637. m_prop_list.push_back(prop);
  2638. return *this;
  2639. }
  2640. void evaluate(size_t index, ValueBase& destination) override;
  2641. std::string description(util::serializer::SerialisationState& state) const override
  2642. {
  2643. std::ostringstream ostr;
  2644. ostr << m_key;
  2645. return ColumnListBase::description(state) + '[' + ostr.str() + ']';
  2646. }
  2647. std::unique_ptr<Subexpr> clone() const override
  2648. {
  2649. return std::unique_ptr<Subexpr>(new ColumnDictionaryKey(*this));
  2650. }
  2651. ColumnDictionaryKey(ColumnDictionaryKey const& other)
  2652. : Columns<Dictionary>(other)
  2653. , m_prop_list(other.m_prop_list)
  2654. {
  2655. init_key(other.m_key);
  2656. }
  2657. private:
  2658. Mixed m_key;
  2659. std::string m_buffer;
  2660. std::vector<std::string> m_prop_list;
  2661. void init_key(Mixed key_value);
  2662. };
  2663. // Returns the keys
  2664. class ColumnDictionaryKeys : public Subexpr2<StringData> {
  2665. public:
  2666. ColumnDictionaryKeys(const Columns<Dictionary>& dict)
  2667. : m_key_type(dict.get_key_type())
  2668. , m_column_key(dict.m_column_key)
  2669. , m_link_map(dict.m_link_map)
  2670. , m_comparison_type(dict.get_comparison_type())
  2671. {
  2672. REALM_ASSERT(m_key_type == type_String);
  2673. }
  2674. ConstTableRef get_base_table() const final
  2675. {
  2676. return m_link_map.get_base_table();
  2677. }
  2678. void set_base_table(ConstTableRef table) final
  2679. {
  2680. m_link_map.set_base_table(table);
  2681. }
  2682. void collect_dependencies(std::vector<TableKey>& tables) const final
  2683. {
  2684. m_link_map.collect_dependencies(tables);
  2685. }
  2686. util::Optional<ExpressionComparisonType> get_comparison_type() const final
  2687. {
  2688. return m_comparison_type;
  2689. }
  2690. void set_cluster(const Cluster* cluster) override;
  2691. void evaluate(size_t index, ValueBase& destination) override;
  2692. std::string description(util::serializer::SerialisationState& state) const override
  2693. {
  2694. return state.describe_expression_type(m_comparison_type) + state.describe_columns(m_link_map, m_column_key) +
  2695. ".@keys";
  2696. }
  2697. std::unique_ptr<Subexpr> clone() const override
  2698. {
  2699. return std::unique_ptr<Subexpr>(new ColumnDictionaryKeys(*this));
  2700. }
  2701. ColumnDictionaryKeys(const ColumnDictionaryKeys& other)
  2702. : m_key_type(other.m_key_type)
  2703. , m_column_key(other.m_column_key)
  2704. , m_link_map(other.m_link_map)
  2705. , m_comparison_type(other.m_comparison_type)
  2706. {
  2707. }
  2708. private:
  2709. DataType m_key_type;
  2710. ColKey m_column_key;
  2711. LinkMap m_link_map;
  2712. util::Optional<ExpressionComparisonType> m_comparison_type;
  2713. // Leaf cache
  2714. using LeafCacheStorage = typename std::aligned_storage<sizeof(ArrayInteger), alignof(Array)>::type;
  2715. using LeafPtr = std::unique_ptr<ArrayInteger, PlacementDelete>;
  2716. LeafCacheStorage m_leaf_cache_storage;
  2717. LeafPtr m_array_ptr;
  2718. ArrayInteger* m_leaf_ptr = nullptr;
  2719. };
  2720. template <typename T>
  2721. class ColumnListSize : public ColumnsCollection<T> {
  2722. public:
  2723. ColumnListSize(const ColumnsCollection<T>& other)
  2724. : ColumnsCollection<T>(other)
  2725. {
  2726. }
  2727. void evaluate(size_t index, ValueBase& destination) override
  2728. {
  2729. if constexpr (realm::is_any_v<T, ObjectId, Int, Bool, UUID>) {
  2730. if (this->m_is_nullable_storage) {
  2731. evaluate<util::Optional<T>>(index, destination);
  2732. return;
  2733. }
  2734. }
  2735. evaluate<T>(index, destination);
  2736. }
  2737. std::unique_ptr<Subexpr> clone() const override
  2738. {
  2739. return std::unique_ptr<Subexpr>(new ColumnListSize<T>(*this));
  2740. }
  2741. private:
  2742. template <typename StorageType>
  2743. void evaluate(size_t index, ValueBase& destination)
  2744. {
  2745. Allocator& alloc = ColumnsCollection<T>::get_alloc();
  2746. Value<int64_t> list_refs;
  2747. this->get_lists(index, list_refs, 1);
  2748. destination.init(list_refs.m_from_list, list_refs.size());
  2749. for (size_t i = 0; i < list_refs.size(); i++) {
  2750. ref_type list_ref = to_ref(list_refs[i].get_int());
  2751. if (list_ref) {
  2752. BPlusTree<StorageType> list(alloc);
  2753. list.init_from_ref(list_ref);
  2754. size_t s = list.size();
  2755. destination.set(i, int64_t(s));
  2756. }
  2757. else {
  2758. destination.set(i, 0);
  2759. }
  2760. }
  2761. }
  2762. };
  2763. template <typename T>
  2764. class ColumnListElementLength : public Subexpr2<Int> {
  2765. public:
  2766. ColumnListElementLength(const ColumnsCollection<T>& source)
  2767. : m_list(source)
  2768. {
  2769. }
  2770. bool has_multiple_values() const override
  2771. {
  2772. return true;
  2773. }
  2774. void evaluate(size_t index, ValueBase& destination) override
  2775. {
  2776. Allocator& alloc = m_list.get_alloc();
  2777. Value<int64_t> list_refs;
  2778. m_list.get_lists(index, list_refs, 1);
  2779. std::vector<Int> sizes;
  2780. for (size_t i = 0; i < list_refs.size(); i++) {
  2781. ref_type list_ref = to_ref(list_refs[i].get_int());
  2782. if (list_ref) {
  2783. BPlusTree<T> list(alloc);
  2784. list.init_from_ref(list_ref);
  2785. const size_t list_size = list.size();
  2786. sizes.reserve(sizes.size() + list_size);
  2787. for (size_t j = 0; j < list_size; j++) {
  2788. if constexpr (std::is_same_v<T, Mixed>) {
  2789. Mixed v = list.get(j);
  2790. if (!v.is_null()) {
  2791. if (v.get_type() == type_String) {
  2792. sizes.push_back(v.get_string().size());
  2793. }
  2794. else if (v.get_type() == type_Binary) {
  2795. sizes.push_back(v.get_binary().size());
  2796. }
  2797. }
  2798. }
  2799. else {
  2800. sizes.push_back(list.get(j).size());
  2801. }
  2802. }
  2803. }
  2804. }
  2805. constexpr bool is_from_list = true;
  2806. destination.init(is_from_list, sizes.size());
  2807. destination.set(sizes.begin(), sizes.end());
  2808. }
  2809. ConstTableRef get_base_table() const override
  2810. {
  2811. return m_list.get_base_table();
  2812. }
  2813. void set_base_table(ConstTableRef table) override
  2814. {
  2815. m_list.set_base_table(table);
  2816. }
  2817. void set_cluster(const Cluster* cluster) override
  2818. {
  2819. m_list.set_cluster(cluster);
  2820. }
  2821. void collect_dependencies(std::vector<TableKey>& tables) const override
  2822. {
  2823. m_list.collect_dependencies(tables);
  2824. }
  2825. std::unique_ptr<Subexpr> clone() const override
  2826. {
  2827. return std::unique_ptr<Subexpr>(new ColumnListElementLength<T>(*this));
  2828. }
  2829. std::string description(util::serializer::SerialisationState& state) const override
  2830. {
  2831. return m_list.description(state) + util::serializer::value_separator + "length";
  2832. }
  2833. util::Optional<ExpressionComparisonType> get_comparison_type() const override
  2834. {
  2835. return m_list.get_comparison_type();
  2836. }
  2837. private:
  2838. ColumnsCollection<T> m_list;
  2839. };
  2840. template <typename T>
  2841. SizeOperator<int64_t> ColumnsCollection<T>::size()
  2842. {
  2843. std::unique_ptr<Subexpr> ptr(new ColumnListSize<T>(*this));
  2844. return SizeOperator<int64_t>(std::move(ptr));
  2845. }
  2846. template <typename T, typename Operation>
  2847. class CollectionColumnAggregate : public Subexpr2<decltype(Operation().result())> {
  2848. public:
  2849. CollectionColumnAggregate(ColumnsCollection<T> column)
  2850. : m_columns_collection(std::move(column))
  2851. {
  2852. if (m_columns_collection.m_column_key.is_dictionary()) {
  2853. m_dictionary_key_type = m_columns_collection.m_link_map.get_target_table()->get_dictionary_key_type(
  2854. m_columns_collection.m_column_key);
  2855. }
  2856. }
  2857. CollectionColumnAggregate(const CollectionColumnAggregate& other)
  2858. : m_columns_collection(other.m_columns_collection)
  2859. , m_dictionary_key_type(other.m_dictionary_key_type)
  2860. {
  2861. }
  2862. DataType get_type() const override
  2863. {
  2864. return DataType(m_columns_collection.m_column_key.get_type());
  2865. }
  2866. std::unique_ptr<Subexpr> clone() const override
  2867. {
  2868. return make_subexpr<CollectionColumnAggregate>(*this);
  2869. }
  2870. ConstTableRef get_base_table() const override
  2871. {
  2872. return m_columns_collection.get_base_table();
  2873. }
  2874. void set_base_table(ConstTableRef table) override
  2875. {
  2876. m_columns_collection.set_base_table(table);
  2877. }
  2878. void set_cluster(const Cluster* cluster) override
  2879. {
  2880. m_columns_collection.set_cluster(cluster);
  2881. }
  2882. void collect_dependencies(std::vector<TableKey>& tables) const override
  2883. {
  2884. m_columns_collection.collect_dependencies(tables);
  2885. }
  2886. void evaluate(size_t index, ValueBase& destination) override
  2887. {
  2888. if (m_dictionary_key_type) {
  2889. if (m_columns_collection.links_exist()) {
  2890. std::vector<ObjKey> links = m_columns_collection.m_link_map.get_links(index);
  2891. auto sz = links.size();
  2892. destination.init_for_links(m_columns_collection.m_link_map.only_unary_links(), sz);
  2893. if (sz == 0 && m_columns_collection.m_link_map.only_unary_links()) {
  2894. set_value_for_empty_dictionary(destination, 0);
  2895. }
  2896. for (size_t t = 0; t < sz; t++) {
  2897. const Obj obj = m_columns_collection.m_link_map.get_target_table()->get_object(links[t]);
  2898. auto dict = obj.get_dictionary(m_columns_collection.m_column_key);
  2899. if (dict.size() > 0) {
  2900. destination.set(t, do_dictionary_agg(dict));
  2901. }
  2902. else {
  2903. set_value_for_empty_dictionary(destination, t);
  2904. }
  2905. }
  2906. }
  2907. else {
  2908. if (auto ref = m_columns_collection.m_leaf_ptr->get(index)) {
  2909. Allocator& alloc = m_columns_collection.get_base_table()->get_alloc();
  2910. Dictionary dict(alloc, m_columns_collection.m_column_key, to_ref(ref));
  2911. destination.set(0, do_dictionary_agg(dict));
  2912. }
  2913. else {
  2914. set_value_for_empty_dictionary(destination, 0);
  2915. }
  2916. }
  2917. }
  2918. else {
  2919. Allocator& alloc = m_columns_collection.get_alloc();
  2920. Value<int64_t> list_refs;
  2921. m_columns_collection.get_lists(index, list_refs, 1);
  2922. size_t sz = list_refs.size();
  2923. REALM_ASSERT_DEBUG(sz > 0 || list_refs.m_from_list);
  2924. // The result is an aggregate value for each table
  2925. destination.init_for_links(!list_refs.m_from_list, sz);
  2926. for (size_t i = 0; i < list_refs.size(); i++) {
  2927. auto list_ref = to_ref(list_refs[i].get_int());
  2928. Operation op;
  2929. if (list_ref) {
  2930. if constexpr (realm::is_any_v<T, ObjectId, Int, Bool, UUID>) {
  2931. if (m_columns_collection.m_is_nullable_storage) {
  2932. accumulate<util::Optional<T>>(op, alloc, list_ref);
  2933. }
  2934. else {
  2935. accumulate<T>(op, alloc, list_ref);
  2936. }
  2937. }
  2938. else {
  2939. accumulate<T>(op, alloc, list_ref);
  2940. }
  2941. }
  2942. if (op.is_null()) {
  2943. destination.set_null(i);
  2944. }
  2945. else {
  2946. destination.set(i, op.result());
  2947. }
  2948. }
  2949. }
  2950. }
  2951. virtual std::string description(util::serializer::SerialisationState& state) const override
  2952. {
  2953. return m_columns_collection.description(state) + util::serializer::value_separator + Operation::description();
  2954. }
  2955. private:
  2956. template <typename StorageType>
  2957. void accumulate(Operation& op, Allocator& alloc, ref_type list_ref)
  2958. {
  2959. BPlusTree<StorageType> list(alloc);
  2960. list.init_from_ref(list_ref);
  2961. size_t s = list.size();
  2962. for (unsigned j = 0; j < s; j++) {
  2963. auto v = list.get(j);
  2964. if (!value_is_null(v)) {
  2965. if constexpr (std::is_same_v<StorageType, util::Optional<T>>) {
  2966. op.accumulate(*v);
  2967. }
  2968. else {
  2969. op.accumulate(v);
  2970. }
  2971. }
  2972. }
  2973. }
  2974. Mixed do_dictionary_agg(const Dictionary& dict)
  2975. {
  2976. if constexpr (std::is_same_v<Operation, aggregate_operations::Maximum<Mixed>>) {
  2977. return *dict.do_max();
  2978. }
  2979. else if constexpr (std::is_same_v<Operation, aggregate_operations::Minimum<Mixed>>) {
  2980. return *dict.do_min();
  2981. }
  2982. else if constexpr (std::is_same_v<Operation, aggregate_operations::Average<Mixed>>) {
  2983. return *dict.do_avg();
  2984. }
  2985. else if constexpr (std::is_same_v<Operation, aggregate_operations::Sum<Mixed>>) {
  2986. return *dict.do_sum();
  2987. }
  2988. REALM_UNREACHABLE();
  2989. }
  2990. inline void set_value_for_empty_dictionary(ValueBase& destination, size_t ndx)
  2991. {
  2992. if constexpr (std::is_same_v<Operation, aggregate_operations::Sum<Mixed>>) {
  2993. destination.set(ndx, 0); // the sum of nothing is zero
  2994. }
  2995. else {
  2996. destination.set_null(ndx);
  2997. }
  2998. }
  2999. ColumnsCollection<T> m_columns_collection;
  3000. util::Optional<DataType> m_dictionary_key_type;
  3001. };
  3002. template <class Operator>
  3003. Query compare(const Subexpr2<Link>& left, const Obj& obj)
  3004. {
  3005. static_assert(std::is_same_v<Operator, Equal> || std::is_same_v<Operator, NotEqual>,
  3006. "Links can only be compared for equality.");
  3007. const Columns<Link>* column = dynamic_cast<const Columns<Link>*>(&left);
  3008. if (column) {
  3009. const LinkMap& link_map = column->link_map();
  3010. REALM_ASSERT(link_map.get_target_table()->get_key() == obj.get_table()->get_key());
  3011. #ifdef REALM_OLDQUERY_FALLBACK
  3012. if (link_map.get_nb_hops() == 1) {
  3013. // We can fall back to Query::links_to for != and == operations on links
  3014. if (link_map.m_link_types[0] == col_type_Link || (link_map.m_link_types[0] == col_type_LinkList)) {
  3015. ConstTableRef t = column->get_base_table();
  3016. Query query(t);
  3017. if (std::is_same_v<Operator, Equal>) {
  3018. return query.equal(link_map.m_link_column_keys[0], obj.get_link());
  3019. }
  3020. else {
  3021. return query.not_equal(link_map.m_link_column_keys[0], obj.get_link());
  3022. }
  3023. }
  3024. }
  3025. #endif
  3026. }
  3027. return make_expression<Compare<Operator>>(left.clone(), make_subexpr<KeyValue>(obj.get_key()));
  3028. }
  3029. template <class Operator>
  3030. Query compare(const Subexpr2<Link>& left, null)
  3031. {
  3032. static_assert(std::is_same_v<Operator, Equal> || std::is_same_v<Operator, NotEqual>,
  3033. "Links can only be compared for equality.");
  3034. return make_expression<Compare<Operator>>(left.clone(), make_subexpr<KeyValue>(ObjKey{}));
  3035. }
  3036. template <class T>
  3037. class Columns : public ObjPropertyExpr<T> {
  3038. constexpr static bool requires_null_column = realm::is_any_v<T, int64_t, bool>;
  3039. public:
  3040. using LeafType = typename ColumnTypeTraits<T>::cluster_leaf_type;
  3041. using NullableLeafType =
  3042. std::conditional_t<requires_null_column, typename ColumnTypeTraits<util::Optional<T>>::cluster_leaf_type,
  3043. LeafType>;
  3044. using ObjPropertyExpr<T>::links_exist;
  3045. using ObjPropertyBase::is_nullable;
  3046. Columns(ColKey column, ConstTableRef table, std::vector<ColKey> links = {},
  3047. util::Optional<ExpressionComparisonType> type = util::none)
  3048. : ObjPropertyExpr<T>(column, table, std::move(links), type)
  3049. {
  3050. }
  3051. Columns(const Columns& other)
  3052. : ObjPropertyExpr<T>(other)
  3053. {
  3054. }
  3055. void set_cluster(const Cluster* cluster) override
  3056. {
  3057. m_array_ptr = nullptr;
  3058. m_leaf_ptr = nullptr;
  3059. if (links_exist()) {
  3060. m_link_map.set_cluster(cluster);
  3061. }
  3062. else if (requires_null_column && is_nullable()) {
  3063. // Create new Leaf
  3064. m_array_ptr = LeafPtr(new (&m_leaf_cache_storage) NullableLeafType(this->get_base_table()->get_alloc()));
  3065. cluster->init_leaf(m_column_key, m_array_ptr.get());
  3066. m_leaf_ptr = m_array_ptr.get();
  3067. }
  3068. else {
  3069. // Create new Leaf
  3070. m_array_ptr = LeafPtr(new (&m_leaf_cache_storage) LeafType(this->get_base_table()->get_alloc()));
  3071. cluster->init_leaf(m_column_key, m_array_ptr.get());
  3072. m_leaf_ptr = m_array_ptr.get();
  3073. }
  3074. }
  3075. template <class LeafType2 = LeafType>
  3076. void evaluate_internal(size_t index, ValueBase& destination)
  3077. {
  3078. using U = typename LeafType2::value_type;
  3079. if (links_exist()) {
  3080. REALM_ASSERT(m_leaf_ptr == nullptr);
  3081. if (m_link_map.only_unary_links()) {
  3082. destination.init(false, 1);
  3083. destination.set_null(0);
  3084. if (auto link_translation_key = m_link_map.get_unary_link_or_not_found(index)) {
  3085. const Obj obj = m_link_map.get_target_table()->get_object(link_translation_key);
  3086. if (!obj.is_null(m_column_key))
  3087. destination.set(0, obj.get<U>(m_column_key));
  3088. }
  3089. }
  3090. else {
  3091. // LinkList with more than 0 values. Create Value with payload for all fields
  3092. std::vector<ObjKey> links = m_link_map.get_links(index);
  3093. destination.init_for_links(m_link_map.only_unary_links(), links.size());
  3094. for (size_t t = 0; t < links.size(); t++) {
  3095. const Obj obj = m_link_map.get_target_table()->get_object(links[t]);
  3096. if (obj.is_null(m_column_key))
  3097. destination.set_null(t);
  3098. else
  3099. destination.set(t, obj.get<U>(m_column_key));
  3100. }
  3101. }
  3102. }
  3103. else {
  3104. REALM_ASSERT(m_leaf_ptr != nullptr);
  3105. auto leaf = static_cast<const LeafType2*>(m_leaf_ptr);
  3106. // Not a Link column
  3107. size_t colsize = leaf->size();
  3108. // Now load `ValueBase::chunk_size` rows from from the leaf into m_storage.
  3109. if constexpr (std::is_same_v<U, int64_t>) {
  3110. // If it's an integer leaf, then it contains the method get_chunk() which copies
  3111. // these values in a super fast way (only feasible if more than chunk_size in column)
  3112. if (index + ValueBase::chunk_size <= colsize) {
  3113. // If you want to modify 'chunk_size' then update Array::get_chunk()
  3114. REALM_ASSERT_3(ValueBase::chunk_size, ==, 8);
  3115. auto leaf_2 = static_cast<const Array*>(leaf);
  3116. int64_t res[ValueBase::chunk_size];
  3117. leaf_2->get_chunk(index, res);
  3118. destination.set(res, res + ValueBase::chunk_size);
  3119. return;
  3120. }
  3121. }
  3122. size_t rows = colsize - index;
  3123. if (rows > ValueBase::chunk_size)
  3124. rows = ValueBase::chunk_size;
  3125. destination.init(false, rows);
  3126. for (size_t t = 0; t < rows; t++) {
  3127. if (leaf->is_null(index + t)) {
  3128. destination.set_null(t);
  3129. }
  3130. else {
  3131. destination.set(t, leaf->get(index + t));
  3132. }
  3133. }
  3134. }
  3135. }
  3136. virtual std::string description(util::serializer::SerialisationState& state) const override
  3137. {
  3138. return state.describe_expression_type(this->m_comparison_type) +
  3139. state.describe_columns(m_link_map, m_column_key);
  3140. }
  3141. // Load values from Column into destination
  3142. void evaluate(size_t index, ValueBase& destination) override
  3143. {
  3144. if (is_nullable()) {
  3145. evaluate_internal<NullableLeafType>(index, destination);
  3146. }
  3147. else {
  3148. evaluate_internal<LeafType>(index, destination);
  3149. }
  3150. }
  3151. void evaluate(ObjKey key, ValueBase& destination) override
  3152. {
  3153. destination.init(false, 1);
  3154. auto table = m_link_map.get_target_table();
  3155. auto obj = table.unchecked_ptr()->get_object(key);
  3156. if (requires_null_column && is_nullable()) {
  3157. destination.set(0, obj.template get<util::Optional<T>>(m_column_key));
  3158. }
  3159. else {
  3160. destination.set(0, obj.template get<T>(m_column_key));
  3161. }
  3162. }
  3163. private:
  3164. using ObjPropertyExpr<T>::m_link_map;
  3165. using ObjPropertyExpr<T>::m_column_key;
  3166. // Leaf cache
  3167. using LeafCacheStorage =
  3168. typename std::aligned_storage<std::max(sizeof(LeafType), sizeof(NullableLeafType)), alignof(LeafType)>::type;
  3169. using LeafPtr = std::unique_ptr<ArrayPayload, PlacementDelete>;
  3170. LeafCacheStorage m_leaf_cache_storage;
  3171. LeafPtr m_array_ptr;
  3172. const ArrayPayload* m_leaf_ptr = nullptr;
  3173. };
  3174. template <typename T, typename Operation>
  3175. class SubColumnAggregate;
  3176. // Defines a uniform interface for aggregation methods.
  3177. class SubColumnBase {
  3178. public:
  3179. virtual std::unique_ptr<Subexpr> max_of() = 0;
  3180. virtual std::unique_ptr<Subexpr> min_of() = 0;
  3181. virtual std::unique_ptr<Subexpr> sum_of() = 0;
  3182. virtual std::unique_ptr<Subexpr> avg_of() = 0;
  3183. };
  3184. template <typename T>
  3185. class SubColumns : public Subexpr, public SubColumnBase {
  3186. public:
  3187. SubColumns(Columns<T>&& column, const LinkMap& link_map)
  3188. : m_column(std::move(column))
  3189. , m_link_map(link_map)
  3190. {
  3191. }
  3192. DataType get_type() const final
  3193. {
  3194. return ColumnTypeTraits<T>::id;
  3195. }
  3196. std::unique_ptr<Subexpr> clone() const override
  3197. {
  3198. return make_subexpr<SubColumns<T>>(*this);
  3199. }
  3200. ConstTableRef get_base_table() const override
  3201. {
  3202. return m_link_map.get_base_table();
  3203. }
  3204. void set_base_table(ConstTableRef table) override
  3205. {
  3206. m_link_map.set_base_table(table);
  3207. m_column.set_base_table(m_link_map.get_target_table());
  3208. }
  3209. void collect_dependencies(std::vector<TableKey>& tables) const override
  3210. {
  3211. m_link_map.collect_dependencies(tables);
  3212. }
  3213. void evaluate(size_t, ValueBase&) override
  3214. {
  3215. // SubColumns can only be used in an expression in conjunction with its aggregate methods.
  3216. REALM_ASSERT(false);
  3217. }
  3218. virtual std::string description(util::serializer::SerialisationState&) const override
  3219. {
  3220. return ""; // by itself there are no conditions, see SubColumnAggregate
  3221. }
  3222. SubColumnAggregate<T, aggregate_operations::Minimum<T>> min() const
  3223. {
  3224. return {m_column, m_link_map};
  3225. }
  3226. SubColumnAggregate<T, aggregate_operations::Maximum<T>> max() const
  3227. {
  3228. return {m_column, m_link_map};
  3229. }
  3230. SubColumnAggregate<T, aggregate_operations::Sum<T>> sum() const
  3231. {
  3232. return {m_column, m_link_map};
  3233. }
  3234. SubColumnAggregate<T, aggregate_operations::Average<T>> average() const
  3235. {
  3236. return {m_column, m_link_map};
  3237. }
  3238. std::unique_ptr<Subexpr> max_of() override
  3239. {
  3240. if constexpr (realm::is_any_v<T, Int, Float, Double, Decimal128, Timestamp>) {
  3241. return max().clone();
  3242. }
  3243. else {
  3244. return {};
  3245. }
  3246. }
  3247. std::unique_ptr<Subexpr> min_of() override
  3248. {
  3249. if constexpr (realm::is_any_v<T, Int, Float, Double, Decimal128, Timestamp>) {
  3250. return min().clone();
  3251. }
  3252. else {
  3253. return {};
  3254. }
  3255. }
  3256. std::unique_ptr<Subexpr> sum_of() override
  3257. {
  3258. if constexpr (realm::is_any_v<T, Int, Float, Double, Decimal128>) {
  3259. return sum().clone();
  3260. }
  3261. else {
  3262. return {};
  3263. }
  3264. }
  3265. std::unique_ptr<Subexpr> avg_of() override
  3266. {
  3267. if constexpr (realm::is_any_v<T, Int, Float, Double, Decimal128>) {
  3268. return average().clone();
  3269. }
  3270. else {
  3271. return {};
  3272. }
  3273. }
  3274. private:
  3275. Columns<T> m_column;
  3276. LinkMap m_link_map;
  3277. };
  3278. template <typename T, typename Operation>
  3279. class SubColumnAggregate : public Subexpr2<decltype(Operation().result())> {
  3280. public:
  3281. SubColumnAggregate(const Columns<T>& column, const LinkMap& link_map)
  3282. : m_column(column)
  3283. , m_link_map(link_map)
  3284. {
  3285. }
  3286. SubColumnAggregate(SubColumnAggregate const& other)
  3287. : m_column(other.m_column)
  3288. , m_link_map(other.m_link_map)
  3289. {
  3290. }
  3291. std::unique_ptr<Subexpr> clone() const override
  3292. {
  3293. return make_subexpr<SubColumnAggregate>(*this);
  3294. }
  3295. ConstTableRef get_base_table() const override
  3296. {
  3297. return m_link_map.get_base_table();
  3298. }
  3299. void set_base_table(ConstTableRef table) override
  3300. {
  3301. m_link_map.set_base_table(table);
  3302. m_column.set_base_table(m_link_map.get_target_table());
  3303. }
  3304. void set_cluster(const Cluster* cluster) override
  3305. {
  3306. m_link_map.set_cluster(cluster);
  3307. }
  3308. void collect_dependencies(std::vector<TableKey>& tables) const override
  3309. {
  3310. m_link_map.collect_dependencies(tables);
  3311. }
  3312. void evaluate(size_t index, ValueBase& destination) override
  3313. {
  3314. std::vector<ObjKey> keys = m_link_map.get_links(index);
  3315. std::sort(keys.begin(), keys.end());
  3316. Operation op;
  3317. for (auto key : keys) {
  3318. Value<T> value;
  3319. m_column.evaluate(key, value);
  3320. size_t value_index = 0;
  3321. if (!value[value_index].is_null()) {
  3322. op.accumulate(value[value_index].template get<T>());
  3323. }
  3324. }
  3325. if (op.is_null()) {
  3326. destination.set_null(0);
  3327. }
  3328. else {
  3329. destination.set(0, op.result());
  3330. }
  3331. }
  3332. virtual std::string description(util::serializer::SerialisationState& state) const override
  3333. {
  3334. util::serializer::SerialisationState empty_state(state.group);
  3335. return state.describe_columns(m_link_map, ColKey()) + util::serializer::value_separator +
  3336. Operation::description() + util::serializer::value_separator + m_column.description(empty_state);
  3337. }
  3338. private:
  3339. Columns<T> m_column;
  3340. LinkMap m_link_map;
  3341. };
  3342. class SubQueryCount : public Subexpr2<Int> {
  3343. public:
  3344. SubQueryCount(const Query& q, const LinkMap& link_map)
  3345. : m_query(q)
  3346. , m_link_map(link_map)
  3347. {
  3348. REALM_ASSERT(q.produces_results_in_table_order());
  3349. REALM_ASSERT(m_query.get_table() == m_link_map.get_target_table());
  3350. }
  3351. ConstTableRef get_base_table() const override
  3352. {
  3353. return m_link_map.get_base_table();
  3354. }
  3355. void set_base_table(ConstTableRef table) override
  3356. {
  3357. m_link_map.set_base_table(table);
  3358. m_query.set_table(m_link_map.get_target_table().cast_away_const());
  3359. }
  3360. void set_cluster(const Cluster* cluster) override
  3361. {
  3362. m_link_map.set_cluster(cluster);
  3363. }
  3364. void collect_dependencies(std::vector<TableKey>& tables) const override
  3365. {
  3366. m_link_map.collect_dependencies(tables);
  3367. }
  3368. void evaluate(size_t index, ValueBase& destination) override
  3369. {
  3370. std::vector<ObjKey> links = m_link_map.get_links(index);
  3371. // std::sort(links.begin(), links.end());
  3372. if (!m_initialized) {
  3373. m_query.init();
  3374. m_initialized = true;
  3375. }
  3376. size_t count = std::accumulate(links.begin(), links.end(), size_t(0), [this](size_t running_count, ObjKey k) {
  3377. const Obj obj = m_link_map.get_target_table()->get_object(k);
  3378. return running_count + m_query.eval_object(obj);
  3379. });
  3380. destination = Value<int64_t>(count);
  3381. }
  3382. virtual std::string description(util::serializer::SerialisationState& state) const override
  3383. {
  3384. REALM_ASSERT(m_link_map.get_base_table() != nullptr);
  3385. std::string target = state.describe_columns(m_link_map, ColKey());
  3386. std::string var_name = state.get_variable_name(m_link_map.get_base_table());
  3387. state.subquery_prefix_list.push_back(var_name);
  3388. std::string desc = "SUBQUERY(" + target + ", " + var_name + ", " + m_query.get_description(state) + ")" +
  3389. util::serializer::value_separator + "@count";
  3390. state.subquery_prefix_list.pop_back();
  3391. return desc;
  3392. }
  3393. std::unique_ptr<Subexpr> clone() const override
  3394. {
  3395. return make_subexpr<SubQueryCount>(*this);
  3396. }
  3397. SubQueryCount(const SubQueryCount& other)
  3398. : m_query(other.m_query)
  3399. , m_link_map(other.m_link_map)
  3400. , m_initialized(false)
  3401. {
  3402. }
  3403. private:
  3404. Query m_query;
  3405. LinkMap m_link_map;
  3406. bool m_initialized = false;
  3407. };
  3408. // The unused template parameter is a hack to avoid a circular dependency between table.hpp and query_expression.hpp.
  3409. template <class>
  3410. class SubQuery {
  3411. public:
  3412. SubQuery(Columns<Link> link_column, Query query)
  3413. : m_query(std::move(query))
  3414. , m_link_map(link_column.link_map())
  3415. {
  3416. REALM_ASSERT(m_link_map.get_target_table() == m_query.get_table());
  3417. }
  3418. SubQueryCount count() const
  3419. {
  3420. return SubQueryCount(m_query, m_link_map);
  3421. }
  3422. private:
  3423. Query m_query;
  3424. LinkMap m_link_map;
  3425. };
  3426. template <class oper>
  3427. class Operator : public Subexpr2<Mixed> {
  3428. public:
  3429. Operator(std::unique_ptr<Subexpr> left, std::unique_ptr<Subexpr> right)
  3430. : m_left(std::move(left))
  3431. , m_right(std::move(right))
  3432. {
  3433. m_left_is_const = m_left->has_single_value();
  3434. m_right_is_const = m_right->has_single_value();
  3435. if (m_left_is_const) {
  3436. m_const_value = m_left->get_mixed();
  3437. }
  3438. else if (m_right_is_const) {
  3439. m_const_value = m_right->get_mixed();
  3440. }
  3441. }
  3442. Operator(const Operator& other)
  3443. : m_left(other.m_left->clone())
  3444. , m_right(other.m_right->clone())
  3445. , m_left_is_const(other.m_left_is_const)
  3446. , m_right_is_const(other.m_right_is_const)
  3447. , m_const_value(other.m_const_value)
  3448. {
  3449. }
  3450. Operator& operator=(const Operator& other)
  3451. {
  3452. if (this != &other) {
  3453. m_left = other.m_left->clone();
  3454. m_right = other.m_right->clone();
  3455. }
  3456. return *this;
  3457. }
  3458. Operator(Operator&&) noexcept = delete;
  3459. Operator& operator=(Operator&&) noexcept = delete;
  3460. DataType get_type() const override
  3461. {
  3462. return m_left->get_type();
  3463. }
  3464. // See comment in base class
  3465. void set_base_table(ConstTableRef table) override
  3466. {
  3467. m_left->set_base_table(table);
  3468. m_right->set_base_table(table);
  3469. }
  3470. void set_cluster(const Cluster* cluster) override
  3471. {
  3472. m_left->set_cluster(cluster);
  3473. m_right->set_cluster(cluster);
  3474. }
  3475. // Recursively fetch tables of columns in expression tree. Used when user first builds a stand-alone expression
  3476. // and
  3477. // binds it to a Query at a later time
  3478. ConstTableRef get_base_table() const override
  3479. {
  3480. ConstTableRef l = m_left->get_base_table();
  3481. ConstTableRef r = m_right->get_base_table();
  3482. // Queries do not support multiple different tables; all tables must be the same.
  3483. REALM_ASSERT(l == nullptr || r == nullptr || l == r);
  3484. // nullptr pointer means expression which isn't yet associated with any table, or is a Value<T>
  3485. return bool(l) ? l : r;
  3486. }
  3487. // destination = operator(left, right)
  3488. void evaluate(size_t index, ValueBase& destination) override
  3489. {
  3490. Value<T> result;
  3491. Value<T> left;
  3492. Value<T> right;
  3493. if (m_left_is_const) {
  3494. m_right->evaluate(index, right);
  3495. result.template fun_const<oper>(m_const_value, right);
  3496. }
  3497. else if (m_right_is_const) {
  3498. m_left->evaluate(index, left);
  3499. result.template fun_const<oper>(left, m_const_value);
  3500. }
  3501. else {
  3502. m_left->evaluate(index, left);
  3503. m_right->evaluate(index, right);
  3504. result.template fun<oper>(left, right);
  3505. }
  3506. destination = result;
  3507. }
  3508. std::string description(util::serializer::SerialisationState& state) const override
  3509. {
  3510. std::string s = "(";
  3511. if (m_left) {
  3512. s += m_left->description(state);
  3513. }
  3514. s += (" " + oper::description() + " ");
  3515. if (m_right) {
  3516. s += m_right->description(state);
  3517. }
  3518. s += ")";
  3519. return s;
  3520. }
  3521. util::Optional<ExpressionComparisonType> get_comparison_type() const override
  3522. {
  3523. if (!m_left_is_const) {
  3524. return m_left->get_comparison_type();
  3525. }
  3526. if (!m_right_is_const) {
  3527. return m_right->get_comparison_type();
  3528. }
  3529. return util::none;
  3530. }
  3531. std::unique_ptr<Subexpr> clone() const override
  3532. {
  3533. return make_subexpr<Operator>(*this);
  3534. }
  3535. private:
  3536. std::unique_ptr<Subexpr> m_left;
  3537. std::unique_ptr<Subexpr> m_right;
  3538. bool m_left_is_const;
  3539. bool m_right_is_const;
  3540. Mixed m_const_value;
  3541. };
  3542. template <class TCond>
  3543. class Compare : public Expression {
  3544. public:
  3545. Compare(std::unique_ptr<Subexpr> left, std::unique_ptr<Subexpr> right)
  3546. : m_left(std::move(left))
  3547. , m_right(std::move(right))
  3548. {
  3549. if (m_left->has_constant_evaluation()) {
  3550. m_left_const_values = dynamic_cast<ValueBase*>(m_left.get());
  3551. }
  3552. if (m_right->has_constant_evaluation()) {
  3553. m_right_const_values = dynamic_cast<ValueBase*>(m_right.get());
  3554. }
  3555. REALM_ASSERT(!(m_left_const_values && m_right_const_values));
  3556. }
  3557. // See comment in base class
  3558. void set_base_table(ConstTableRef table) override
  3559. {
  3560. m_left->set_base_table(table);
  3561. m_right->set_base_table(table);
  3562. }
  3563. void set_cluster(const Cluster* cluster) override
  3564. {
  3565. if (m_has_matches) {
  3566. m_cluster = cluster;
  3567. }
  3568. else {
  3569. m_left->set_cluster(cluster);
  3570. m_right->set_cluster(cluster);
  3571. }
  3572. }
  3573. double init() override
  3574. {
  3575. double dT = 50.0;
  3576. if ((m_left->has_single_value()) || (m_right->has_single_value())) {
  3577. dT = 10.0;
  3578. if constexpr (std::is_same_v<TCond, Equal>) {
  3579. // If the property not being constant has a search index we can speed things up by
  3580. // finding all matches up front.
  3581. Mixed const_value;
  3582. Subexpr* column;
  3583. if (m_left->has_single_value()) {
  3584. const_value = m_left->get_mixed();
  3585. column = m_right.get();
  3586. }
  3587. else {
  3588. const_value = m_right->get_mixed();
  3589. column = m_left.get();
  3590. }
  3591. if (column->has_search_index() &&
  3592. column->get_comparison_type().value_or(ExpressionComparisonType::Any) ==
  3593. ExpressionComparisonType::Any) {
  3594. if (const_value.is_null()) {
  3595. const ObjPropertyBase* prop = dynamic_cast<const ObjPropertyBase*>(m_right.get());
  3596. // when checking for null across links, null links are considered matches,
  3597. // so we must compute the slow matching even if there is an index.
  3598. if (!prop || prop->links_exist()) {
  3599. return dT;
  3600. }
  3601. else {
  3602. m_matches = column->find_all(Mixed());
  3603. }
  3604. }
  3605. else {
  3606. if (column->get_type() != const_value.get_type()) {
  3607. // If the type we are looking for is not the same type as the target
  3608. // column, we cannot use the index
  3609. return dT;
  3610. }
  3611. m_matches = column->find_all(const_value);
  3612. }
  3613. // Sort
  3614. std::sort(m_matches.begin(), m_matches.end());
  3615. // Remove all duplicates
  3616. m_matches.erase(std::unique(m_matches.begin(), m_matches.end()), m_matches.end());
  3617. m_has_matches = true;
  3618. m_index_get = 0;
  3619. m_index_end = m_matches.size();
  3620. dT = 0;
  3621. }
  3622. }
  3623. }
  3624. return dT;
  3625. }
  3626. // Recursively fetch tables of columns in expression tree. Used when user first builds a stand-alone expression
  3627. // and binds it to a Query at a later time
  3628. ConstTableRef get_base_table() const override
  3629. {
  3630. ConstTableRef l = m_left->get_base_table();
  3631. ConstTableRef r = m_right->get_base_table();
  3632. // All main tables in each subexpression of a query (table.columns() or table.link()) must be the same.
  3633. REALM_ASSERT(l == nullptr || r == nullptr || l == r);
  3634. // nullptr pointer means expression which isn't yet associated with any table, or is a Value<T>
  3635. return (l) ? l : r;
  3636. }
  3637. void collect_dependencies(std::vector<TableKey>& tables) const override
  3638. {
  3639. m_left->collect_dependencies(tables);
  3640. m_right->collect_dependencies(tables);
  3641. }
  3642. size_t find_first(size_t start, size_t end) const override
  3643. {
  3644. if (m_has_matches) {
  3645. if (m_index_end == 0 || start >= end)
  3646. return not_found;
  3647. ObjKey first_key = m_cluster->get_real_key(start);
  3648. ObjKey actual_key;
  3649. // Sequential lookup optimization: when the query isn't constrained
  3650. // to a LnkLst we'll get find_first() requests in ascending order,
  3651. // so we can do a simple linear scan.
  3652. if (m_index_get < m_index_end && m_matches[m_index_get] <= first_key) {
  3653. actual_key = m_matches[m_index_get];
  3654. // skip through keys which are in "earlier" leafs than the one selected by start..end:
  3655. while (first_key > actual_key) {
  3656. m_index_get++;
  3657. if (m_index_get == m_index_end)
  3658. return not_found;
  3659. actual_key = m_matches[m_index_get];
  3660. }
  3661. }
  3662. // Otherwise if we get requests out of order we have to do a more
  3663. // expensive binary search
  3664. else {
  3665. auto it = std::lower_bound(m_matches.begin(), m_matches.end(), first_key);
  3666. if (it == m_matches.end())
  3667. return not_found;
  3668. actual_key = *it;
  3669. }
  3670. // if actual key is bigger than last key, it is not in this leaf
  3671. ObjKey last_key = start + 1 == end ? first_key : m_cluster->get_real_key(end - 1);
  3672. if (actual_key > last_key)
  3673. return not_found;
  3674. // key is known to be in this leaf, so find key whithin leaf keys
  3675. return m_cluster->lower_bound_key(ObjKey(actual_key.value - m_cluster->get_offset()));
  3676. }
  3677. size_t match;
  3678. ValueBase left_buf;
  3679. ValueBase right_buf;
  3680. const util::Optional<ExpressionComparisonType> left_cmp_type = m_left->get_comparison_type();
  3681. const util::Optional<ExpressionComparisonType> right_cmp_type = m_right->get_comparison_type();
  3682. ValueBase* left = m_left_const_values ? m_left_const_values : &left_buf;
  3683. ValueBase* right = m_right_const_values ? m_right_const_values : &right_buf;
  3684. for (; start < end;) {
  3685. if (!m_left_const_values)
  3686. m_left->evaluate(start, left_buf);
  3687. if (!m_right_const_values)
  3688. m_right->evaluate(start, right_buf);
  3689. match = ValueBase::template compare<TCond>(*left, *right, left_cmp_type, right_cmp_type);
  3690. if (match != not_found && match + start < end)
  3691. return start + match;
  3692. size_t rows = (left->m_from_list || right->m_from_list) ? 1 : minimum(right->size(), left->size());
  3693. start += rows;
  3694. }
  3695. return not_found; // no match
  3696. }
  3697. virtual std::string description(util::serializer::SerialisationState& state) const override
  3698. {
  3699. if constexpr (realm::is_any_v<TCond, BeginsWith, BeginsWithIns, EndsWith, EndsWithIns, Contains, ContainsIns,
  3700. Like, LikeIns>) {
  3701. // these string conditions have the arguments reversed but the order is important
  3702. // operations ==, and != can be reversed because the produce the same results both ways
  3703. return util::serializer::print_value(m_right->description(state) + " " + TCond::description() + " " +
  3704. m_left->description(state));
  3705. }
  3706. else {
  3707. state.target_table = m_right->get_target_table();
  3708. std::string ret = m_left->description(state) + " " + TCond::description() + " ";
  3709. state.target_table = m_left->get_target_table();
  3710. ret += m_right->description(state);
  3711. return ret;
  3712. }
  3713. }
  3714. std::unique_ptr<Expression> clone() const override
  3715. {
  3716. return std::unique_ptr<Expression>(new Compare(*this));
  3717. }
  3718. private:
  3719. Compare(const Compare& other)
  3720. : m_left(other.m_left->clone())
  3721. , m_right(other.m_right->clone())
  3722. {
  3723. if (m_left->has_constant_evaluation()) {
  3724. m_left_const_values = dynamic_cast<ValueBase*>(m_left.get());
  3725. }
  3726. if (m_right->has_constant_evaluation()) {
  3727. m_right_const_values = dynamic_cast<ValueBase*>(m_right.get());
  3728. }
  3729. }
  3730. std::unique_ptr<Subexpr> m_left;
  3731. std::unique_ptr<Subexpr> m_right;
  3732. const Cluster* m_cluster;
  3733. ValueBase* m_left_const_values = nullptr;
  3734. ValueBase* m_right_const_values = nullptr;
  3735. bool m_has_matches = false;
  3736. std::vector<ObjKey> m_matches;
  3737. mutable size_t m_index_get = 0;
  3738. size_t m_index_end = 0;
  3739. };
  3740. } // namespace realm
  3741. #endif // REALM_QUERY_EXPRESSION_HPP