You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

query_test.rb 123KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312
  1. # frozen_string_literal: true
  2. # Redmine - project management software
  3. # Copyright (C) 2006- Jean-Philippe Lang
  4. #
  5. # This program is free software; you can redistribute it and/or
  6. # modify it under the terms of the GNU General Public License
  7. # as published by the Free Software Foundation; either version 2
  8. # of the License, or (at your option) any later version.
  9. #
  10. # This program is distributed in the hope that it will be useful,
  11. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. # GNU General Public License for more details.
  14. #
  15. # You should have received a copy of the GNU General Public License
  16. # along with this program; if not, write to the Free Software
  17. # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  18. require_relative '../test_helper'
  19. class QueryTest < ActiveSupport::TestCase
  20. include Redmine::I18n
  21. fixtures :projects, :enabled_modules, :users, :user_preferences, :members,
  22. :member_roles, :roles, :trackers, :issue_statuses,
  23. :issue_categories, :enumerations, :issues,
  24. :watchers, :custom_fields, :custom_values, :versions,
  25. :queries,
  26. :projects_trackers,
  27. :custom_fields_trackers,
  28. :workflows, :journals,
  29. :attachments, :time_entries
  30. def setup
  31. User.current = nil
  32. end
  33. def test_query_with_roles_visibility_should_validate_roles
  34. set_language_if_valid 'en'
  35. query = IssueQuery.new(:name => 'Query', :visibility => IssueQuery::VISIBILITY_ROLES)
  36. assert !query.save
  37. assert_include "Roles cannot be blank", query.errors.full_messages
  38. query.role_ids = [1, 2]
  39. assert query.save
  40. end
  41. def test_changing_roles_visibility_should_clear_roles
  42. query = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_ROLES, :role_ids => [1, 2])
  43. assert_equal 2, query.roles.count
  44. query.visibility = IssueQuery::VISIBILITY_PUBLIC
  45. query.save!
  46. assert_equal 0, query.roles.count
  47. end
  48. def test_available_filters_should_be_ordered
  49. set_language_if_valid 'en'
  50. query = IssueQuery.new
  51. assert_equal 0, query.available_filters.keys.index('status_id')
  52. expected_order = [
  53. "Status",
  54. "Project",
  55. "Tracker",
  56. "Priority"
  57. ]
  58. assert_equal expected_order,
  59. (query.available_filters.values.pluck(:name) & expected_order)
  60. end
  61. def test_available_filters_with_custom_fields_should_be_ordered
  62. set_language_if_valid 'en'
  63. UserCustomField.create!(
  64. :name => 'order test', :field_format => 'string',
  65. :is_for_all => true, :is_filter => true
  66. )
  67. query = IssueQuery.new
  68. expected_order = [
  69. "Searchable field",
  70. "Database",
  71. "Project's Development status",
  72. "Author's order test",
  73. "Assignee's order test"
  74. ]
  75. assert_equal expected_order,
  76. (query.available_filters.values.pluck(:name) & expected_order)
  77. end
  78. def test_custom_fields_for_all_projects_should_be_available_in_global_queries
  79. query = IssueQuery.new(:project => nil, :name => '_')
  80. assert query.available_filters.has_key?('cf_1')
  81. assert !query.available_filters.has_key?('cf_3')
  82. end
  83. def test_system_shared_versions_should_be_available_in_global_queries
  84. Version.find(2).update_attribute :sharing, 'system'
  85. query = IssueQuery.new(:project => nil, :name => '_')
  86. assert query.available_filters.has_key?('fixed_version_id')
  87. assert query.available_filters['fixed_version_id'][:values].detect {|v| v[1] == '2'}
  88. end
  89. def test_project_filter_in_global_queries
  90. query = IssueQuery.new(:project => nil, :name => '_')
  91. project_filter = query.available_filters["project_id"]
  92. assert_not_nil project_filter
  93. project_ids = project_filter[:values].pluck(1)
  94. assert project_ids.include?("1") # public project
  95. assert !project_ids.include?("2") # private project user cannot see
  96. end
  97. def test_available_filters_should_not_include_fields_disabled_on_all_trackers
  98. Tracker.all.each do |tracker|
  99. tracker.core_fields = Tracker::CORE_FIELDS - ['start_date']
  100. tracker.save!
  101. end
  102. query = IssueQuery.new(:name => '_')
  103. assert_include 'due_date', query.available_filters
  104. assert_not_include 'start_date', query.available_filters
  105. end
  106. def test_filter_values_without_project_should_be_arrays
  107. q = IssueQuery.new
  108. assert_nil q.project
  109. q.available_filters.each do |name, filter|
  110. values = filter.values
  111. assert (values.nil? || values.is_a?(Array)),
  112. "#values for #{name} filter returned a #{values.class.name}"
  113. end
  114. end
  115. def test_filter_values_with_project_should_be_arrays
  116. q = IssueQuery.new(:project => Project.find(1))
  117. assert_not_nil q.project
  118. q.available_filters.each do |name, filter|
  119. values = filter.values
  120. assert (values.nil? || values.is_a?(Array)),
  121. "#values for #{name} filter returned a #{values.class.name}"
  122. end
  123. end
  124. def find_issues_with_query(query)
  125. Issue.joins(:status, :tracker, :project, :priority).
  126. where(query.statement).to_a
  127. end
  128. def assert_find_issues_with_query_is_successful(query)
  129. assert_nothing_raised do
  130. find_issues_with_query(query)
  131. end
  132. end
  133. def assert_query_statement_includes(query, condition)
  134. assert_include condition, query.statement
  135. end
  136. def assert_query_result(expected, query)
  137. assert_nothing_raised do
  138. assert_equal expected.map(&:id).sort, query.issues.map(&:id).sort
  139. assert_equal expected.size, query.issue_count
  140. end
  141. end
  142. def test_query_should_allow_shared_versions_for_a_project_query
  143. subproject_version = Version.find(4)
  144. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  145. filter = query.available_filters["fixed_version_id"]
  146. assert_not_nil filter
  147. assert_include subproject_version.id.to_s, filter[:values].map(&:second)
  148. end
  149. def test_query_with_multiple_custom_fields
  150. query = IssueQuery.find(1)
  151. assert query.valid?
  152. issues = find_issues_with_query(query)
  153. assert_equal 1, issues.length
  154. assert_equal Issue.find(3), issues.first
  155. end
  156. def test_operator_none
  157. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  158. query.add_filter('fixed_version_id', '!*', [''])
  159. query.add_filter('cf_1', '!*', [''])
  160. assert query.statement.include?("#{Issue.table_name}.fixed_version_id IS NULL")
  161. assert query.statement.include?("#{CustomValue.table_name}.value IS NULL OR #{CustomValue.table_name}.value = ''")
  162. find_issues_with_query(query)
  163. end
  164. def test_operator_none_for_integer
  165. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  166. query.add_filter('estimated_hours', '!*', [''])
  167. issues = find_issues_with_query(query)
  168. assert !issues.empty?
  169. assert issues.all? {|i| !i.estimated_hours}
  170. end
  171. def test_operator_none_for_date
  172. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  173. query.add_filter('start_date', '!*', [''])
  174. issues = find_issues_with_query(query)
  175. assert !issues.empty?
  176. assert issues.all? {|i| i.start_date.nil?}
  177. end
  178. def test_operator_none_for_string_custom_field
  179. CustomField.find(2).update_attribute :default_value, ""
  180. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  181. query.add_filter('cf_2', '!*', [''])
  182. assert query.has_filter?('cf_2')
  183. issues = find_issues_with_query(query)
  184. assert !issues.empty?
  185. assert issues.all? {|i| i.custom_field_value(2).blank?}
  186. end
  187. def test_operator_none_for_blank_text
  188. query = IssueQuery.new(:name => '_')
  189. query.add_filter('status_id', '*', [''])
  190. query.add_filter('description', '!*', [''])
  191. assert query.has_filter?('description')
  192. issues = find_issues_with_query(query)
  193. assert issues.any?
  194. assert issues.all? {|i| i.description.blank?}
  195. assert_equal [11, 12], issues.map(&:id).sort
  196. end
  197. def test_operator_any_for_blank_text
  198. Issue.where(id: [1, 2]).update_all(description: '')
  199. query = IssueQuery.new(:name => '_')
  200. query.add_filter('status_id', '*', [''])
  201. query.add_filter('description', '*', [''])
  202. assert query.has_filter?('description')
  203. issues = find_issues_with_query(query)
  204. assert issues.any?
  205. assert issues.all? {|i| i.description.present?}
  206. assert_empty issues.map(&:id) & [1, 2]
  207. end
  208. def test_operator_all
  209. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  210. query.add_filter('fixed_version_id', '*', [''])
  211. query.add_filter('cf_1', '*', [''])
  212. assert query.statement.include?("#{Issue.table_name}.fixed_version_id IS NOT NULL")
  213. assert query.statement.include?("#{CustomValue.table_name}.value IS NOT NULL AND #{CustomValue.table_name}.value <> ''")
  214. find_issues_with_query(query)
  215. end
  216. def test_operator_all_for_date
  217. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  218. query.add_filter('start_date', '*', [''])
  219. issues = find_issues_with_query(query)
  220. assert !issues.empty?
  221. assert issues.all? {|i| i.start_date.present?}
  222. end
  223. def test_operator_all_for_string_custom_field
  224. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  225. query.add_filter('cf_2', '*', [''])
  226. assert query.has_filter?('cf_2')
  227. issues = find_issues_with_query(query)
  228. assert !issues.empty?
  229. assert issues.all? {|i| i.custom_field_value(2).present?}
  230. end
  231. def test_numeric_filter_should_not_accept_non_numeric_values
  232. query = IssueQuery.new(:name => '_')
  233. query.add_filter('estimated_hours', '=', ['a'])
  234. assert query.has_filter?('estimated_hours')
  235. assert !query.valid?
  236. end
  237. def test_operator_is_on_float
  238. Issue.where(:id => 2).update_all("estimated_hours = 171.2")
  239. query = IssueQuery.new(:name => '_')
  240. query.add_filter('estimated_hours', '=', ['171.20'])
  241. issues = find_issues_with_query(query)
  242. assert_equal 1, issues.size
  243. assert_equal 2, issues.first.id
  244. end
  245. def test_operator_is_on_issue_id_should_accept_comma_separated_values
  246. query = IssueQuery.new(:name => '_')
  247. query.add_filter("issue_id", '=', ['1,3'])
  248. issues = find_issues_with_query(query)
  249. assert_equal 2, issues.size
  250. assert_equal [1, 3], issues.map(&:id).sort
  251. end
  252. def test_operator_is_on_parent_id_should_accept_comma_separated_values
  253. Issue.where(:id => [2, 4]).update_all(:parent_id => 1)
  254. Issue.where(:id => 5).update_all(:parent_id => 3)
  255. query = IssueQuery.new(:name => '_')
  256. query.add_filter("parent_id", '=', ['1,3'])
  257. issues = find_issues_with_query(query)
  258. assert_equal 3, issues.size
  259. assert_equal [2, 4, 5], issues.map(&:id).sort
  260. end
  261. def test_operator_is_on_child_id_should_accept_comma_separated_values
  262. Issue.where(:id => [2, 4]).update_all(:parent_id => 1)
  263. Issue.where(:id => 5).update_all(:parent_id => 3)
  264. query = IssueQuery.new(:name => '_')
  265. query.add_filter("child_id", '=', ['2,4,5'])
  266. issues = find_issues_with_query(query)
  267. assert_equal 2, issues.size
  268. assert_equal [1, 3], issues.map(&:id).sort
  269. end
  270. def test_operator_between_on_issue_id_should_return_range
  271. query = IssueQuery.new(:name => '_')
  272. query.add_filter("issue_id", '><', ['2', '3'])
  273. issues = find_issues_with_query(query)
  274. assert_equal 2, issues.size
  275. assert_equal [2, 3], issues.map(&:id).sort
  276. end
  277. def test_operator_is_on_integer_custom_field
  278. f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_for_all => true, :is_filter => true, :trackers => Tracker.all)
  279. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7')
  280. CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12')
  281. CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
  282. query = IssueQuery.new(:name => '_')
  283. query.add_filter("cf_#{f.id}", '=', ['12'])
  284. issues = find_issues_with_query(query)
  285. assert_equal 1, issues.size
  286. assert_equal 2, issues.first.id
  287. end
  288. def test_operator_is_on_integer_custom_field_should_accept_negative_value
  289. f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_for_all => true, :is_filter => true, :trackers => Tracker.all)
  290. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7')
  291. CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '-12')
  292. CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
  293. query = IssueQuery.new(:name => '_')
  294. query.add_filter("cf_#{f.id}", '=', ['-12'])
  295. assert query.valid?
  296. issues = find_issues_with_query(query)
  297. assert_equal 1, issues.size
  298. assert_equal 2, issues.first.id
  299. end
  300. def test_operator_is_on_float_custom_field
  301. f = IssueCustomField.create!(:name => 'filter', :field_format => 'float', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
  302. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7.3')
  303. CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12.7')
  304. CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
  305. query = IssueQuery.new(:name => '_')
  306. query.add_filter("cf_#{f.id}", '=', ['12.7'])
  307. issues = find_issues_with_query(query)
  308. assert_equal 1, issues.size
  309. assert_equal 2, issues.first.id
  310. end
  311. def test_operator_is_on_float_custom_field_should_accept_negative_value
  312. f = IssueCustomField.create!(:name => 'filter', :field_format => 'float', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
  313. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7.3')
  314. CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '-12.7')
  315. CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
  316. query = IssueQuery.new(:name => '_')
  317. query.add_filter("cf_#{f.id}", '=', ['-12.7'])
  318. assert query.valid?
  319. issues = find_issues_with_query(query)
  320. assert_equal 1, issues.size
  321. assert_equal 2, issues.first.id
  322. end
  323. def test_operator_is_on_multi_list_custom_field
  324. f = IssueCustomField.create!(:name => 'filter', :field_format => 'list', :is_filter => true, :is_for_all => true,
  325. :possible_values => ['value1', 'value2', 'value3'], :multiple => true, :trackers => Tracker.all)
  326. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value1')
  327. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value2')
  328. CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => 'value1')
  329. query = IssueQuery.new(:name => '_')
  330. query.add_filter("cf_#{f.id}", '=', ['value1'])
  331. issues = find_issues_with_query(query)
  332. assert_equal [1, 3], issues.map(&:id).sort
  333. query = IssueQuery.new(:name => '_')
  334. query.add_filter("cf_#{f.id}", '=', ['value2'])
  335. issues = find_issues_with_query(query)
  336. assert_equal [1], issues.map(&:id).sort
  337. end
  338. def test_operator_is_not_on_multi_list_custom_field
  339. f = IssueCustomField.create!(:name => 'filter', :field_format => 'list', :is_filter => true, :is_for_all => true,
  340. :possible_values => ['value1', 'value2', 'value3'], :multiple => true, :trackers => Tracker.all)
  341. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value1')
  342. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value2')
  343. CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => 'value1')
  344. query = IssueQuery.new(:name => '_')
  345. query.add_filter("cf_#{f.id}", '!', ['value1'])
  346. issues = find_issues_with_query(query)
  347. assert !issues.map(&:id).include?(1)
  348. assert !issues.map(&:id).include?(3)
  349. query = IssueQuery.new(:name => '_')
  350. query.add_filter("cf_#{f.id}", '!', ['value2'])
  351. issues = find_issues_with_query(query)
  352. assert !issues.map(&:id).include?(1)
  353. assert issues.map(&:id).include?(3)
  354. end
  355. def test_operator_is_on_string_custom_field_with_utf8_value
  356. f = IssueCustomField.create!(:name => 'filter', :field_format => 'string', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
  357. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'Kiểm')
  358. query = IssueQuery.new(:name => '_')
  359. query.add_filter("cf_#{f.id}", '=', ['Kiểm'])
  360. issues = find_issues_with_query(query)
  361. assert_equal [1], issues.map(&:id).sort
  362. end
  363. def test_operator_is_on_is_private_field
  364. # is_private filter only available for those who can set issues private
  365. User.current = User.find(2)
  366. query = IssueQuery.new(:name => '_')
  367. assert query.available_filters.key?('is_private')
  368. query.add_filter("is_private", '=', ['1'])
  369. issues = find_issues_with_query(query)
  370. assert issues.any?
  371. assert_nil issues.detect {|issue| !issue.is_private?}
  372. end
  373. def test_operator_is_not_on_is_private_field
  374. # is_private filter only available for those who can set issues private
  375. User.current = User.find(2)
  376. query = IssueQuery.new(:name => '_')
  377. assert query.available_filters.key?('is_private')
  378. query.add_filter("is_private", '!', ['1'])
  379. issues = find_issues_with_query(query)
  380. assert issues.any?
  381. assert_nil issues.detect {|issue| issue.is_private?}
  382. end
  383. def test_operator_greater_than
  384. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  385. query.add_filter('done_ratio', '>=', ['40'])
  386. assert query.statement.include?("#{Issue.table_name}.done_ratio >= 40.0")
  387. find_issues_with_query(query)
  388. end
  389. def test_operator_greater_than_a_float
  390. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  391. query.add_filter('estimated_hours', '>=', ['40.5'])
  392. assert query.statement.include?("#{Issue.table_name}.estimated_hours >= 40.5")
  393. find_issues_with_query(query)
  394. end
  395. def test_operator_greater_than_on_int_custom_field
  396. f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
  397. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7')
  398. CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12')
  399. CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
  400. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  401. query.add_filter("cf_#{f.id}", '>=', ['8'])
  402. issues = find_issues_with_query(query)
  403. assert_equal 1, issues.size
  404. assert_equal 2, issues.first.id
  405. end
  406. def test_operator_lesser_than
  407. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  408. query.add_filter('done_ratio', '<=', ['30'])
  409. assert query.statement.include?("#{Issue.table_name}.done_ratio <= 30.0")
  410. find_issues_with_query(query)
  411. end
  412. def test_operator_lesser_than_on_custom_field
  413. f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
  414. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  415. query.add_filter("cf_#{f.id}", '<=', ['30'])
  416. assert_match /CAST.+ <= 30\.0/, query.statement
  417. find_issues_with_query(query)
  418. end
  419. def test_operator_lesser_than_on_date_custom_field
  420. f = IssueCustomField.create!(:name => 'filter', :field_format => 'date', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
  421. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '2013-04-11')
  422. CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '2013-05-14')
  423. CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
  424. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  425. query.add_filter("cf_#{f.id}", '<=', ['2013-05-01'])
  426. issue_ids = find_issues_with_query(query).map(&:id)
  427. assert_include 1, issue_ids
  428. assert_not_include 2, issue_ids
  429. assert_not_include 3, issue_ids
  430. end
  431. def test_operator_between
  432. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  433. query.add_filter('done_ratio', '><', ['30', '40'])
  434. assert_include "#{Issue.table_name}.done_ratio BETWEEN 30.0 AND 40.0", query.statement
  435. find_issues_with_query(query)
  436. end
  437. def test_operator_between_on_custom_field
  438. f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
  439. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  440. query.add_filter("cf_#{f.id}", '><', ['30', '40'])
  441. assert_match /CAST.+ BETWEEN 30.0 AND 40.0/, query.statement
  442. find_issues_with_query(query)
  443. end
  444. def test_time_entry_operator_is_on_issue_parent_id_should_accept_comma_separated_values
  445. issue1 = Issue.generate!(project_id: 'ecookbook', parent_id: 2)
  446. entry1 = TimeEntry.generate!(issue: issue1)
  447. issue2 = Issue.generate!(project_id: 'ecookbook', parent_id: 5)
  448. entry2 = TimeEntry.generate!(issue: issue2)
  449. query = TimeEntryQuery.new(:name => '_')
  450. query.add_filter("issue.parent_id", '=', ['2,5'])
  451. entries = TimeEntry.where(query.statement).to_a
  452. assert_equal 2, entries.size
  453. assert_equal [entry1.id, entry2.id].sort, entries.map(&:id).sort
  454. end
  455. def test_time_entry_contains_operator_is_on_issue_parent_id
  456. issue1 = Issue.generate!(project_id: 'ecookbook', parent_id: 2)
  457. entry1 = TimeEntry.generate!(issue: issue1)
  458. issue2 = Issue.generate!(project_id: 'ecookbook', parent_id: issue1.id)
  459. entry2 = TimeEntry.generate!(issue: issue2)
  460. query = TimeEntryQuery.new(:name => '_')
  461. query.add_filter("issue.parent_id", '~', ['2'])
  462. entries = TimeEntry.where(query.statement).to_a
  463. assert_equal 2, entries.size
  464. assert_equal [entry1.id, entry2.id].sort, entries.map(&:id).sort
  465. end
  466. def test_date_filter_should_not_accept_non_date_values
  467. query = IssueQuery.new(:name => '_')
  468. query.add_filter('created_on', '=', ['a'])
  469. assert query.has_filter?('created_on')
  470. assert !query.valid?
  471. end
  472. def test_date_filter_should_not_accept_invalid_date_values
  473. query = IssueQuery.new(:name => '_')
  474. query.add_filter('created_on', '=', ['2011-01-34'])
  475. assert query.has_filter?('created_on')
  476. assert !query.valid?
  477. end
  478. def test_relative_date_filter_should_not_accept_non_integer_values
  479. query = IssueQuery.new(:name => '_')
  480. query.add_filter('created_on', '>t-', ['a'])
  481. assert query.has_filter?('created_on')
  482. assert !query.valid?
  483. end
  484. def test_operator_date_equals
  485. query = IssueQuery.new(:name => '_')
  486. query.add_filter('due_date', '=', ['2011-07-10'])
  487. assert_match /issues\.due_date > '#{quoted_date "2011-07-09"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-07-10"} 23:59:59(\.\d+)?/,
  488. query.statement
  489. find_issues_with_query(query)
  490. end
  491. def test_operator_date_lesser_than
  492. query = IssueQuery.new(:name => '_')
  493. query.add_filter('due_date', '<=', ['2011-07-10'])
  494. assert_match /issues\.due_date <= '#{quoted_date "2011-07-10"} 23:59:59(\.\d+)?/, query.statement
  495. find_issues_with_query(query)
  496. end
  497. def test_operator_date_lesser_than_with_timestamp
  498. query = IssueQuery.new(:name => '_')
  499. query.add_filter('updated_on', '<=', ['2011-07-10T19:13:52'])
  500. assert_match /issues\.updated_on <= '#{quoted_date "2011-07-10"} 19:13:52/, query.statement
  501. find_issues_with_query(query)
  502. end
  503. def test_operator_date_greater_than
  504. query = IssueQuery.new(:name => '_')
  505. query.add_filter('due_date', '>=', ['2011-07-10'])
  506. assert_match /issues\.due_date > '#{quoted_date "2011-07-09"} 23:59:59(\.\d+)?'/, query.statement
  507. find_issues_with_query(query)
  508. end
  509. def test_operator_date_greater_than_with_timestamp
  510. query = IssueQuery.new(:name => '_')
  511. query.add_filter('updated_on', '>=', ['2011-07-10T19:13:52'])
  512. assert_match /issues\.updated_on > '#{quoted_date "2011-07-10"} 19:13:51(\.0+)?'/, query.statement
  513. find_issues_with_query(query)
  514. end
  515. def test_operator_date_between
  516. query = IssueQuery.new(:name => '_')
  517. query.add_filter('due_date', '><', ['2011-06-23', '2011-07-10'])
  518. assert_match /issues\.due_date > '#{quoted_date "2011-06-22"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-07-10"} 23:59:59(\.\d+)?'/,
  519. query.statement
  520. find_issues_with_query(query)
  521. end
  522. def test_operator_in_more_than
  523. Issue.find(7).update_attribute(:due_date, (Date.today + 15))
  524. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  525. query.add_filter('due_date', '>t+', ['15'])
  526. issues = find_issues_with_query(query)
  527. assert !issues.empty?
  528. issues.each {|issue| assert(issue.due_date >= (Date.today + 15))}
  529. end
  530. def test_operator_in_less_than
  531. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  532. query.add_filter('due_date', '<t+', ['15'])
  533. issues = find_issues_with_query(query)
  534. assert !issues.empty?
  535. issues.each {|issue| assert(issue.due_date <= (Date.today + 15))}
  536. end
  537. def test_operator_in_the_next_days
  538. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  539. query.add_filter('due_date', '><t+', ['15'])
  540. issues = find_issues_with_query(query)
  541. assert !issues.empty?
  542. issues.each {|issue| assert(issue.due_date >= Date.today && issue.due_date <= (Date.today + 15))}
  543. end
  544. def test_operator_less_than_ago
  545. Issue.find(7).update_attribute(:due_date, (Date.today - 3))
  546. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  547. query.add_filter('due_date', '>t-', ['3'])
  548. issues = find_issues_with_query(query)
  549. assert !issues.empty?
  550. issues.each {|issue| assert(issue.due_date >= (Date.today - 3))}
  551. end
  552. def test_operator_in_the_past_days
  553. Issue.find(7).update_attribute(:due_date, (Date.today - 3))
  554. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  555. query.add_filter('due_date', '><t-', ['3'])
  556. issues = find_issues_with_query(query)
  557. assert !issues.empty?
  558. issues.each {|issue| assert(issue.due_date >= (Date.today - 3) && issue.due_date <= Date.today)}
  559. end
  560. def test_operator_more_than_ago
  561. Issue.find(7).update_attribute(:due_date, (Date.today - 10))
  562. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  563. query.add_filter('due_date', '<t-', ['10'])
  564. assert query.statement.include?("#{Issue.table_name}.due_date <=")
  565. issues = find_issues_with_query(query)
  566. assert !issues.empty?
  567. issues.each {|issue| assert(issue.due_date <= (Date.today - 10))}
  568. end
  569. def test_operator_in
  570. Issue.find(7).update_attribute(:due_date, (Date.today + 2))
  571. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  572. query.add_filter('due_date', 't+', ['2'])
  573. issues = find_issues_with_query(query)
  574. assert !issues.empty?
  575. issues.each {|issue| assert_equal((Date.today + 2), issue.due_date)}
  576. end
  577. def test_operator_ago
  578. Issue.find(7).update_attribute(:due_date, (Date.today - 3))
  579. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  580. query.add_filter('due_date', 't-', ['3'])
  581. issues = find_issues_with_query(query)
  582. assert !issues.empty?
  583. issues.each {|issue| assert_equal((Date.today - 3), issue.due_date)}
  584. end
  585. def test_operator_today
  586. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  587. query.add_filter('due_date', 't', [''])
  588. issues = find_issues_with_query(query)
  589. assert !issues.empty?
  590. issues.each {|issue| assert_equal Date.today, issue.due_date}
  591. end
  592. def test_operator_tomorrow
  593. issue = Issue.generate!(:due_date => User.current.today.tomorrow)
  594. other_issues = []
  595. other_issues << Issue.generate!(:due_date => User.current.today.yesterday)
  596. other_issues << Issue.generate!(:due_date => User.current.today + 2)
  597. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  598. query.add_filter('due_date', 'nd', [''])
  599. issues = find_issues_with_query(query)
  600. assert_include issue, issues
  601. other_issues.each {|i| assert_not_include i, issues}
  602. end
  603. def test_operator_date_periods
  604. %w(t ld w lw l2w m lm y nd nw nm).each do |operator|
  605. query = IssueQuery.new(:name => '_')
  606. query.add_filter('due_date', operator, [''])
  607. assert query.valid?
  608. assert query.issues
  609. end
  610. end
  611. def test_operator_datetime_periods
  612. %w(t ld w lw l2w m lm y).each do |operator|
  613. query = IssueQuery.new(:name => '_')
  614. query.add_filter('created_on', operator, [''])
  615. assert query.valid?
  616. assert query.issues
  617. end
  618. end
  619. def test_operator_contains
  620. issue = Issue.generate!(:subject => 'AbCdEfG')
  621. query = IssueQuery.new(:name => '_')
  622. query.add_filter('subject', '~', ['cdeF'])
  623. result = find_issues_with_query(query)
  624. assert_include issue, result
  625. result.each {|issue| assert issue.subject.downcase.include?('cdef')}
  626. end
  627. def test_operator_contains_with_utf8_string
  628. issue = Issue.generate!(:subject => 'Subject contains Kiểm')
  629. query = IssueQuery.new(:name => '_')
  630. query.add_filter('subject', '~', ['Kiểm'])
  631. result = find_issues_with_query(query)
  632. assert_include issue, result
  633. assert_equal 1, result.size
  634. end
  635. def test_operator_does_not_contain
  636. issue = Issue.generate!(:subject => 'AbCdEfG')
  637. query = IssueQuery.new(:name => '_')
  638. query.add_filter('subject', '!~', ['cdeF'])
  639. result = find_issues_with_query(query)
  640. assert_not_include issue, result
  641. end
  642. def test_operator_contains_any_of
  643. User.current = User.find(1)
  644. query = IssueQuery.new(
  645. :name => '_',
  646. :filters => {
  647. 'subject' => {
  648. :operator => '*~',
  649. :values => ['close block']
  650. }
  651. }
  652. )
  653. result = find_issues_with_query(query)
  654. assert_equal [8, 9, 10, 11, 12], result.map(&:id).sort
  655. result.each {|issue| assert issue.subject =~ /(close|block)/i}
  656. end
  657. def test_operator_contains_any_of_with_any_searchable_text
  658. User.current = User.find(1)
  659. query = IssueQuery.new(
  660. :name => '_',
  661. :filters => {
  662. 'any_searchable' => {
  663. :operator => '*~',
  664. :values => ['recipe categories']
  665. }
  666. }
  667. )
  668. result = find_issues_with_query(query)
  669. assert_equal [1, 2, 3], result.map(&:id).sort
  670. end
  671. def test_operator_contains_any_of_with_attachment
  672. User.current = User.find(1)
  673. query = IssueQuery.new(
  674. :name => '_',
  675. :filters => {
  676. 'attachment' => {
  677. :operator => '*~',
  678. :values => ['source changeset']
  679. }
  680. }
  681. )
  682. result = find_issues_with_query(query)
  683. assert_equal [2, 3], result.map(&:id).sort
  684. end
  685. def test_operator_contsins_any_of_with_attachment_description
  686. User.current = User.find(1)
  687. query = IssueQuery.new(
  688. :name => '_',
  689. :filters => {
  690. 'attachment_description' => {
  691. :operator => '*~',
  692. :values => ['ruby issue']
  693. }
  694. }
  695. )
  696. result = find_issues_with_query(query)
  697. assert_equal [2, 14], result.map(&:id).sort
  698. end
  699. def test_operator_changed_from
  700. User.current = User.find(1)
  701. issue1 = Issue.find(2)
  702. issue1.init_journal(User.current)
  703. issue1.update(status_id: 1) # Assigned (2) -> New
  704. issue2 = Issue.find(8)
  705. issue2.init_journal(User.current)
  706. issue2.update(status_id: 2) # Closed (5) -> Assigned
  707. query = IssueQuery.new(
  708. :name => '_',
  709. :filters => {
  710. 'status_id' => {
  711. :operator => 'cf',
  712. :values => [2, 5] # Assigned, Closed
  713. }
  714. }
  715. )
  716. result = find_issues_with_query(query)
  717. assert_equal(
  718. [[2, 'New'], [8, 'Assigned']],
  719. result.sort_by(&:id).map {|issue| [issue.id, issue.status.name]}
  720. )
  721. end
  722. def test_operator_has_been
  723. User.current = User.find(1)
  724. issue = Issue.find(8)
  725. issue.init_journal(User.current)
  726. issue.update(status_id: 2) # Closed (5) -> Assigned
  727. query = IssueQuery.new(
  728. :name => '_',
  729. :filters => {
  730. 'status_id' => {
  731. :operator => 'ev',
  732. :values => [5] # Closed
  733. }
  734. }
  735. )
  736. result = find_issues_with_query(query)
  737. assert_equal(
  738. [[8, 'Assigned'], [11, 'Closed'], [12, 'Closed']],
  739. result.sort_by(&:id).map {|issue| [issue.id, issue.status.name]}
  740. )
  741. end
  742. def test_operator_has_never_been
  743. User.current = User.find(1)
  744. issue = Issue.find(8)
  745. issue.init_journal(User.current)
  746. issue.update(status_id: 2) # Closed (5) -> Assigned
  747. query = IssueQuery.new(
  748. :name => '_',
  749. :filters => {
  750. 'status_id' => {
  751. :operator => '!ev',
  752. :values => [5] # Closed
  753. }
  754. }
  755. )
  756. result = find_issues_with_query(query)
  757. expected = Issue.order(:id).ids - [8, 11, 12]
  758. assert_equal expected, result.map(&:id).sort
  759. end
  760. def test_range_for_this_week_with_week_starting_on_monday
  761. I18n.locale = :fr
  762. assert_equal '1', I18n.t(:general_first_day_of_week)
  763. Date.stubs(:today).returns(Date.parse('2011-04-29'))
  764. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  765. query.add_filter('due_date', 'w', [''])
  766. assert_match /issues\.due_date > '#{quoted_date "2011-04-24"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-05-01"} 23:59:59(\.\d+)?/,
  767. query.statement
  768. I18n.locale = :en
  769. end
  770. def test_range_for_this_week_with_week_starting_on_sunday
  771. I18n.locale = :en
  772. assert_equal '7', I18n.t(:general_first_day_of_week)
  773. Date.stubs(:today).returns(Date.parse('2011-04-29'))
  774. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  775. query.add_filter('due_date', 'w', [''])
  776. assert_match /issues\.due_date > '#{quoted_date "2011-04-23"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-04-30"} 23:59:59(\.\d+)?/,
  777. query.statement
  778. end
  779. def test_range_for_next_week_with_week_starting_on_monday
  780. I18n.locale = :fr
  781. assert_equal '1', I18n.t(:general_first_day_of_week)
  782. Date.stubs(:today).returns(Date.parse('2011-04-29')) # Friday
  783. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  784. query.add_filter('due_date', 'nw', [''])
  785. assert_match /issues\.due_date > '#{quoted_date "2011-05-01"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-05-08"} 23:59:59(\.\d+)?/,
  786. query.statement
  787. I18n.locale = :en
  788. end
  789. def test_range_for_next_week_with_week_starting_on_sunday
  790. I18n.locale = :en
  791. assert_equal '7', I18n.t(:general_first_day_of_week)
  792. Date.stubs(:today).returns(Date.parse('2011-04-29')) # Friday
  793. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  794. query.add_filter('due_date', 'nw', [''])
  795. assert_match /issues\.due_date > '#{quoted_date "2011-04-30"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-05-07"} 23:59:59(\.\d+)?/,
  796. query.statement
  797. end
  798. def test_range_for_next_month
  799. Date.stubs(:today).returns(Date.parse('2011-04-29')) # Friday
  800. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  801. query.add_filter('due_date', 'nm', [''])
  802. assert_match /issues\.due_date > '#{quoted_date "2011-04-30"} 23:59:59(\.\d+)?' AND issues\.due_date <= '#{quoted_date "2011-05-31"} 23:59:59(\.\d+)?/,
  803. query.statement
  804. end
  805. def test_filter_assigned_to_me
  806. user = User.find(2)
  807. group = Group.find(10)
  808. group.users << user
  809. other_group = Group.find(11)
  810. Member.create!(:project_id => 1, :principal => group, :role_ids => [1])
  811. Member.create!(:project_id => 1, :principal => other_group, :role_ids => [1])
  812. User.current = user
  813. with_settings :issue_group_assignment => '1' do
  814. i1 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => user)
  815. i2 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => group)
  816. i3 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => other_group)
  817. query =
  818. IssueQuery.new(
  819. :name => '_',
  820. :filters => {
  821. 'assigned_to_id' => {
  822. :operator => '=',
  823. :values => ['me']
  824. }
  825. }
  826. )
  827. result = query.issues
  828. assert_equal(
  829. Issue.visible.where(:assigned_to_id => ([2] + user.reload.group_ids)).sort_by(&:id),
  830. result.sort_by(&:id)
  831. )
  832. assert result.include?(i1)
  833. assert result.include?(i2)
  834. assert !result.include?(i3)
  835. end
  836. end
  837. def test_filter_notes
  838. user = User.generate!
  839. Journal.create!(:user_id => user.id, :journalized => Issue.find(2), :notes => 'Notes.')
  840. Journal.create!(:user_id => user.id, :journalized => Issue.find(3), :notes => 'Notes.')
  841. issue_journals = Issue.find(1).journals.sort
  842. assert_equal ['Journal notes', 'Some notes with Redmine links: #2, r2.'], issue_journals.map(&:notes)
  843. assert_equal [false, false], issue_journals.map(&:private_notes)
  844. query = IssueQuery.new(:name => '_')
  845. filter_name = 'notes'
  846. assert_include filter_name, query.available_filters.keys
  847. {
  848. '~' => [1, 2, 3],
  849. '!~' => Issue.ids.sort - [1, 2, 3],
  850. '^' => [2, 3],
  851. '$' => [1],
  852. }.each do |operator, expected|
  853. query.filters = {filter_name => {:operator => operator, :values => ['Notes']}}
  854. assert_equal expected, find_issues_with_query(query).map(&:id).sort
  855. end
  856. end
  857. def test_filter_notes_should_ignore_private_notes_that_are_not_visible
  858. user = User.generate!
  859. Journal.create!(:user_id => user.id, :journalized => Issue.find(2), :notes => 'Notes.', :private_notes => true)
  860. Journal.create!(:user_id => user.id, :journalized => Issue.find(3), :notes => 'Notes.')
  861. issue_journals = Issue.find(1).journals.sort
  862. assert_equal ['Journal notes', 'Some notes with Redmine links: #2, r2.'], issue_journals.map(&:notes)
  863. assert_equal [false, false], issue_journals.map(&:private_notes)
  864. query = IssueQuery.new(:name => '_')
  865. filter_name = 'notes'
  866. assert_include filter_name, query.available_filters.keys
  867. query.filters = {filter_name => {:operator => '~', :values => ['Notes']}}
  868. assert_equal [1, 3], find_issues_with_query(query).map(&:id).sort
  869. end
  870. def test_filter_any_searchable
  871. User.current = User.find(1)
  872. query = IssueQuery.new(
  873. :name => '_',
  874. :filters => {
  875. 'any_searchable' => {
  876. :operator => '~',
  877. :values => ['recipe']
  878. }
  879. }
  880. )
  881. result = find_issues_with_query(query)
  882. assert_equal [1, 2, 3], result.map(&:id).sort
  883. end
  884. def test_filter_any_searchable_with_multiple_words
  885. User.current = User.find(1)
  886. query = IssueQuery.new(
  887. :name => '_',
  888. :filters => {
  889. 'any_searchable' => {
  890. :operator => '~',
  891. :values => ['recipe categories']
  892. }
  893. }
  894. )
  895. result = find_issues_with_query(query)
  896. assert_equal [2], result.map(&:id)
  897. end
  898. def test_filter_any_searchable_with_multiple_words_negative
  899. User.current = User.find(1)
  900. query_result_ids = ->(op, value) do
  901. query = IssueQuery.new(
  902. :name => '_',
  903. :filters => {'any_searchable' => {:operator => op, :values => [value]}}
  904. )
  905. find_issues_with_query(query).map(&:id).sort
  906. end
  907. ids = query_result_ids.call('!~', 'recipe categories')
  908. ids_word1 = query_result_ids.call('~', 'recipe')
  909. ids_word2 = query_result_ids.call('~', 'categories')
  910. # Neither "recipe" nor "categories" are in the subject, description,
  911. # notes, etc.
  912. assert ids, Issue.ids.sort - ids_word1 - ids_word2
  913. end
  914. def test_filter_any_searchable_no_matches
  915. User.current = User.find(1)
  916. query = IssueQuery.new(
  917. :name => '_',
  918. :filters => {
  919. 'any_searchable' => {
  920. :operator => '~',
  921. :values => ['SomethingThatDoesNotExist']
  922. }
  923. }
  924. )
  925. result = find_issues_with_query(query)
  926. assert_empty result.map(&:id)
  927. end
  928. def test_filter_any_searchable_negative
  929. User.current = User.find(1)
  930. query = IssueQuery.new(
  931. :name => '_',
  932. :filters => {
  933. 'any_searchable' => {
  934. :operator => '!~',
  935. :values => ['recipe']
  936. }
  937. }
  938. )
  939. result = find_issues_with_query(query)
  940. assert_not_includes [1, 2, 3], result.map(&:id)
  941. end
  942. def test_filter_any_searchable_negative_no_matches
  943. User.current = User.find(1)
  944. query = IssueQuery.new(
  945. :name => '_',
  946. :filters => {
  947. 'any_searchable' => {
  948. :operator => '!~',
  949. :values => ['SomethingThatDoesNotExist']
  950. }
  951. }
  952. )
  953. result = find_issues_with_query(query)
  954. assert_not_empty result.map(&:id)
  955. end
  956. def test_filter_any_searchable_should_search_searchable_custom_fields
  957. User.current = User.find(1)
  958. query = IssueQuery.new(
  959. :name => '_',
  960. :filters => {
  961. 'any_searchable' => {
  962. :operator => '~',
  963. :values => ['125']
  964. }
  965. }
  966. )
  967. result = find_issues_with_query(query)
  968. assert_equal [1, 3], result.map(&:id).sort
  969. end
  970. def test_filter_any_searchable_with_my_projects
  971. # This user's project is ecookbook only
  972. User.current = User.find_by(login: 'dlopper')
  973. query = IssueQuery.new(
  974. :name => '_',
  975. :filters => {
  976. 'any_searchable' => {:operator => '~', :values => ['issue']},
  977. 'project_id' => {:operator => '=', :values => ['mine']}
  978. }
  979. )
  980. result = find_issues_with_query(query)
  981. assert_equal [7, 8, 11, 12], result.map(&:id).sort
  982. result.each {|issue| assert_equal 1, issue.project_id}
  983. end
  984. def test_filter_any_searchable_with_my_bookmarks
  985. # This user bookmarks two projects, ecookbook and private-child
  986. User.current = User.find(1)
  987. query = IssueQuery.new(
  988. :name => '_',
  989. :filters => {
  990. 'any_searchable' => {:operator => '~', :values => ['issue']},
  991. 'project_id' => {:operator => '=', :values => ['bookmarks']}
  992. }
  993. )
  994. result = find_issues_with_query(query)
  995. assert_equal [6, 7, 8, 9, 10, 11, 12], result.map(&:id).sort
  996. result.each {|issue| assert_includes [1, 5], issue.project_id}
  997. end
  998. def test_filter_any_searchable_with_open_issues_should_search_only_open_issues
  999. User.current = User.find(1)
  1000. query = IssueQuery.new(
  1001. :name => '_',
  1002. :filters => {
  1003. 'status_id' => {:operator => 'o'}
  1004. }
  1005. )
  1006. result = query.sql_for_any_searchable_field(nil, '~', ['issue'])
  1007. assert_match /issues.id IN \([\d,]+\)/, result
  1008. ids = result.scan(/\d+/).map(&:to_i).sort
  1009. assert_equal [4, 5, 6, 7, 9, 10, 13, 14], ids
  1010. end
  1011. def test_filter_updated_by
  1012. user = User.generate!
  1013. Journal.create!(:user_id => user.id, :journalized => Issue.find(2), :notes => 'Notes')
  1014. Journal.create!(:user_id => user.id, :journalized => Issue.find(3), :notes => 'Notes')
  1015. Journal.create!(:user_id => 2, :journalized => Issue.find(3), :notes => 'Notes')
  1016. query = IssueQuery.new(:name => '_')
  1017. filter_name = "updated_by"
  1018. assert_include filter_name, query.available_filters.keys
  1019. query.filters = {filter_name => {:operator => '=', :values => [user.id]}}
  1020. assert_equal [2, 3], find_issues_with_query(query).map(&:id).sort
  1021. query.filters = {filter_name => {:operator => '!', :values => [user.id]}}
  1022. assert_equal (Issue.ids.sort - [2, 3]), find_issues_with_query(query).map(&:id).sort
  1023. end
  1024. def test_filter_updated_by_should_ignore_private_notes_that_are_not_visible
  1025. user = User.generate!
  1026. Journal.create!(:user_id => user.id, :journalized => Issue.find(2), :notes => 'Notes', :private_notes => true)
  1027. Journal.create!(:user_id => user.id, :journalized => Issue.find(3), :notes => 'Notes')
  1028. query = IssueQuery.new(:name => '_')
  1029. filter_name = "updated_by"
  1030. assert_include filter_name, query.available_filters.keys
  1031. with_current_user User.anonymous do
  1032. query.filters = {filter_name => {:operator => '=', :values => [user.id]}}
  1033. assert_equal [3], find_issues_with_query(query).map(&:id).sort
  1034. end
  1035. end
  1036. def test_filter_updated_by_me
  1037. user = User.generate!
  1038. Journal.create!(:user_id => user.id, :journalized => Issue.find(2), :notes => 'Notes')
  1039. with_current_user user do
  1040. query = IssueQuery.new(:name => '_')
  1041. filter_name = "updated_by"
  1042. assert_include filter_name, query.available_filters.keys
  1043. query.filters = {filter_name => {:operator => '=', :values => ['me']}}
  1044. assert_equal [2], find_issues_with_query(query).map(&:id).sort
  1045. end
  1046. end
  1047. def test_filter_last_updated_by
  1048. user = User.generate!
  1049. Journal.create!(:user_id => user.id, :journalized => Issue.find(2), :notes => 'Notes')
  1050. Journal.create!(:user_id => user.id, :journalized => Issue.find(3), :notes => 'Notes')
  1051. Journal.create!(:user_id => 2, :journalized => Issue.find(3), :notes => 'Notes')
  1052. query = IssueQuery.new(:name => '_')
  1053. filter_name = "last_updated_by"
  1054. assert_include filter_name, query.available_filters.keys
  1055. query.filters = {filter_name => {:operator => '=', :values => [user.id]}}
  1056. assert_equal [2], find_issues_with_query(query).map(&:id).sort
  1057. end
  1058. def test_filter_last_updated_by_should_ignore_private_notes_that_are_not_visible
  1059. user1 = User.generate!
  1060. user2 = User.generate!
  1061. Journal.create!(:user_id => user1.id, :journalized => Issue.find(2), :notes => 'Notes')
  1062. Journal.create!(:user_id => user2.id, :journalized => Issue.find(2), :notes => 'Notes', :private_notes => true)
  1063. query = IssueQuery.new(:name => '_')
  1064. filter_name = "last_updated_by"
  1065. assert_include filter_name, query.available_filters.keys
  1066. with_current_user User.anonymous do
  1067. query.filters = {filter_name => {:operator => '=', :values => [user1.id]}}
  1068. assert_equal [2], find_issues_with_query(query).map(&:id).sort
  1069. query.filters = {filter_name => {:operator => '=', :values => [user2.id]}}
  1070. assert_equal [], find_issues_with_query(query).map(&:id).sort
  1071. end
  1072. with_current_user User.find(2) do
  1073. query.filters = {filter_name => {:operator => '=', :values => [user1.id]}}
  1074. assert_equal [], find_issues_with_query(query).map(&:id).sort
  1075. query.filters = {filter_name => {:operator => '=', :values => [user2.id]}}
  1076. assert_equal [2], find_issues_with_query(query).map(&:id).sort
  1077. end
  1078. end
  1079. def test_user_custom_field_filtered_on_me
  1080. User.current = User.find(2)
  1081. cf = IssueCustomField.create!(:field_format => 'user', :is_for_all => true, :is_filter => true, :name => 'User custom field', :tracker_ids => [1])
  1082. issue1 = Issue.create!(:project_id => 1, :tracker_id => 1, :custom_field_values => {cf.id.to_s => '2'}, :subject => 'Test', :author_id => 1)
  1083. issue2 = Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {cf.id.to_s => '3'})
  1084. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  1085. filter = query.available_filters["cf_#{cf.id}"]
  1086. assert_not_nil filter
  1087. assert_include 'me', filter[:values].pluck(1)
  1088. query.filters = {"cf_#{cf.id}" => {:operator => '=', :values => ['me']}}
  1089. result = query.issues
  1090. assert_equal 1, result.size
  1091. assert_equal issue1, result.first
  1092. end
  1093. def test_filter_on_chained_user_custom_field
  1094. user = User.find(2)
  1095. User.current = user
  1096. user_cf = UserCustomField.find(4)
  1097. user_cf.update! is_filter: true
  1098. issue_cf = IssueCustomField.create!(:field_format => 'user', :is_for_all => true, :is_filter => true, :name => 'User custom field', :tracker_ids => [1])
  1099. issue1 = Issue.create!(:project_id => 1, :tracker_id => 1, :custom_field_values => {issue_cf.id.to_s => '2'}, :subject => 'Test', :author_id => 1)
  1100. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  1101. query.filters = {"cf_#{issue_cf.id}.cf_#{user_cf.id}" => {:operator => '~', :values => ['01 42']}}
  1102. result = query.issues
  1103. assert_equal 1, result.size
  1104. assert_equal issue1, result.first
  1105. end
  1106. def test_filter_on_chained_user_custom_field_of_type_float
  1107. user_cf = UserCustomField.find(5)
  1108. user_cf.update! is_filter: true
  1109. issue_cf = IssueCustomField.create!(:field_format => 'user', :is_for_all => true, :is_filter => true, :name => 'User custom field', :tracker_ids => [1])
  1110. issue1 = Issue.create!(:project_id => 1, :tracker_id => 1, :custom_field_values => {issue_cf.id.to_s => '2'}, :subject => 'Test', :author_id => 1)
  1111. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  1112. query.filters = {"cf_#{issue_cf.id}.cf_#{user_cf.id}" => {:operator => '=', :values => ["30.1"]}}
  1113. assert query.issues
  1114. end
  1115. def test_filter_on_me_by_anonymous_user
  1116. User.current = nil
  1117. query =
  1118. IssueQuery.new(
  1119. :name => '_',
  1120. :filters => {
  1121. 'assigned_to_id' => {
  1122. :operator => '=',
  1123. :values => ['me']
  1124. }
  1125. }
  1126. )
  1127. assert_equal [], query.issues
  1128. end
  1129. def test_filter_my_projects
  1130. User.current = User.find(2)
  1131. query = IssueQuery.new(:name => '_')
  1132. filter = query.available_filters['project_id']
  1133. assert_not_nil filter
  1134. assert_include 'mine', filter[:values].pluck(1)
  1135. query.filters = {'project_id' => {:operator => '=', :values => ['mine']}}
  1136. result = query.issues
  1137. assert_nil result.detect {|issue| !User.current.member_of?(issue.project)}
  1138. end
  1139. def test_filter_my_bookmarks
  1140. User.current = User.find(1)
  1141. query = ProjectQuery.new(:name => '_')
  1142. filter = query.available_filters['id']
  1143. assert_not_nil filter
  1144. assert_include 'bookmarks', filter[:values].pluck(1)
  1145. query.filters = {'id' => {:operator => '=', :values => ['bookmarks']}}
  1146. result = query.results_scope
  1147. assert_equal [1, 5], result.map(&:id).sort
  1148. end
  1149. def test_filter_my_bookmarks_for_user_without_bookmarked_projects
  1150. User.current = User.find(2)
  1151. query = ProjectQuery.new(:name => '_')
  1152. filter = query.available_filters['id']
  1153. assert_not_include 'bookmarks', filter[:values].pluck(1)
  1154. end
  1155. def test_filter_project_parent_id_with_my_projects
  1156. User.current = User.find(1)
  1157. query = ProjectQuery.new(:name => '_')
  1158. filter = query.available_filters['parent_id']
  1159. assert_not_nil filter
  1160. assert_include 'mine', filter[:values].pluck(1)
  1161. query.filters = {'parent_id' => {:operator => '=', :values => ['mine']}}
  1162. result = query.results_scope
  1163. my_projects = User.current.memberships.map(&:project_id)
  1164. assert_equal Project.where(parent_id: my_projects).ids, result.map(&:id).sort
  1165. end
  1166. def test_filter_project_parent_id_with_my_bookmarks
  1167. User.current = User.find(1)
  1168. query = ProjectQuery.new(:name => '_')
  1169. filter = query.available_filters['parent_id']
  1170. assert_not_nil filter
  1171. assert_include 'bookmarks', filter[:values].pluck(1)
  1172. query.filters = {'parent_id' => {:operator => '=', :values => ['bookmarks']}}
  1173. result = query.results_scope
  1174. bookmarks = User.current.bookmarked_project_ids
  1175. assert_equal Project.where(parent_id: bookmarks).ids.sort, result.map(&:id).sort
  1176. end
  1177. def test_filter_watched_issues
  1178. User.current = User.find(1)
  1179. query =
  1180. IssueQuery.new(
  1181. :name => '_',
  1182. :filters => {
  1183. 'watcher_id' => {
  1184. :operator => '=',
  1185. :values => ['me']
  1186. }
  1187. }
  1188. )
  1189. result = find_issues_with_query(query)
  1190. assert_not_nil result
  1191. assert !result.empty?
  1192. assert_equal Issue.visible.watched_by(User.current).sort_by(&:id), result.sort_by(&:id)
  1193. end
  1194. def test_filter_watched_issues_with_groups_also
  1195. user = User.find(2)
  1196. group = Group.find(10)
  1197. group.users << user
  1198. Issue.find(3).add_watcher(user)
  1199. Issue.find(7).add_watcher(group)
  1200. User.current = user
  1201. query =
  1202. IssueQuery.new(
  1203. :name => '_',
  1204. :filters => {
  1205. 'watcher_id' => {
  1206. :operator => '=',
  1207. :values => ['me']
  1208. }
  1209. }
  1210. )
  1211. result = find_issues_with_query(query)
  1212. assert_not_nil result
  1213. assert !result.empty?
  1214. assert_equal [3, 7], result.sort_by(&:id).pluck(:id)
  1215. end
  1216. def test_filter_unwatched_issues
  1217. User.current = User.find(1)
  1218. query =
  1219. IssueQuery.new(
  1220. :name => '_',
  1221. :filters => {
  1222. 'watcher_id' => {
  1223. :operator => '!', :values => ['me']
  1224. }
  1225. }
  1226. )
  1227. result = find_issues_with_query(query)
  1228. assert_not_nil result
  1229. assert !result.empty?
  1230. assert_equal((Issue.visible - Issue.watched_by(User.current)).sort_by(&:id).size, result.sort_by(&:id).size)
  1231. end
  1232. def test_filter_on_watched_issues_with_view_issue_watchers_permission
  1233. User.current = User.find(1)
  1234. User.current.admin = true
  1235. assert User.current.allowed_to?(:view_issue_watchers, Project.find(1))
  1236. Issue.find(1).add_watcher User.current
  1237. Issue.find(3).add_watcher User.find(3)
  1238. query =
  1239. IssueQuery.new(
  1240. :name => '_',
  1241. :filters => {
  1242. 'watcher_id' => {
  1243. :operator => '=',
  1244. :values => ['me', '3']
  1245. }
  1246. }
  1247. )
  1248. result = find_issues_with_query(query)
  1249. assert_includes result, Issue.find(1)
  1250. assert_includes result, Issue.find(3)
  1251. end
  1252. def test_filter_on_watched_issues_without_view_issue_watchers_permission
  1253. User.current = User.find(1)
  1254. User.current.admin = false
  1255. assert !User.current.allowed_to?(:view_issue_watchers, Project.find(1))
  1256. Issue.find(1).add_watcher User.current
  1257. Issue.find(3).add_watcher User.find(3)
  1258. query =
  1259. IssueQuery.new(
  1260. :name => '_',
  1261. :filters => {
  1262. 'watcher_id' => {
  1263. :operator => '=',
  1264. :values => ['me', '3']
  1265. }
  1266. }
  1267. )
  1268. result = find_issues_with_query(query)
  1269. assert_includes result, Issue.find(1)
  1270. assert_not_includes result, Issue.find(3)
  1271. end
  1272. def test_filter_on_custom_field_should_ignore_projects_with_field_disabled
  1273. field =
  1274. IssueCustomField.generate!(
  1275. :trackers => Tracker.all, :project_ids => [1, 3, 5],
  1276. :is_for_all => false, :is_filter => true
  1277. )
  1278. Issue.generate!(:project_id => 3, :tracker_id => 2,
  1279. :custom_field_values => {field.id.to_s => 'Foo'})
  1280. Issue.generate!(:project_id => 5, :tracker_id => 2,
  1281. :custom_field_values => {field.id.to_s => 'Foo'})
  1282. User.current = User.find(1)
  1283. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  1284. query.filters = {"cf_#{field.id}" => {:operator => '=', :values => ['Foo']}}
  1285. assert_equal 2, find_issues_with_query(query).size
  1286. field.project_ids = [1, 3] # Disable the field for project 4
  1287. field.save!
  1288. assert_equal 1, find_issues_with_query(query).size
  1289. end
  1290. def test_filter_on_custom_field_should_ignore_trackers_with_field_disabled
  1291. field = IssueCustomField.generate!(:tracker_ids => [1, 2], :is_for_all => true, :is_filter => true)
  1292. Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {field.id.to_s => 'Foo'})
  1293. Issue.generate!(:project_id => 1, :tracker_id => 2, :custom_field_values => {field.id.to_s => 'Foo'})
  1294. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  1295. query.filters = {"cf_#{field.id}" => {:operator => '=', :values => ['Foo']}}
  1296. assert_equal 2, find_issues_with_query(query).size
  1297. field.tracker_ids = [1] # Disable the field for tracker 2
  1298. field.save!
  1299. assert_equal 1, find_issues_with_query(query).size
  1300. end
  1301. def test_filter_on_project_custom_field
  1302. field = ProjectCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
  1303. CustomValue.create!(:custom_field => field, :customized => Project.find(3), :value => 'Foo')
  1304. CustomValue.create!(:custom_field => field, :customized => Project.find(5), :value => 'Foo')
  1305. query = IssueQuery.new(:name => '_')
  1306. filter_name = "project.cf_#{field.id}"
  1307. assert_include filter_name, query.available_filters.keys
  1308. query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
  1309. assert_equal [3, 5], find_issues_with_query(query).map(&:project_id).uniq.sort
  1310. end
  1311. def test_filter_on_author_custom_field
  1312. field = UserCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
  1313. CustomValue.create!(:custom_field => field, :customized => User.find(3), :value => 'Foo')
  1314. query = IssueQuery.new(:name => '_')
  1315. filter_name = "author.cf_#{field.id}"
  1316. assert_include filter_name, query.available_filters.keys
  1317. query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
  1318. assert_equal [3], find_issues_with_query(query).map(&:author_id).uniq.sort
  1319. end
  1320. def test_filter_on_assigned_to_custom_field
  1321. field = UserCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
  1322. CustomValue.create!(:custom_field => field, :customized => User.find(3), :value => 'Foo')
  1323. query = IssueQuery.new(:name => '_')
  1324. filter_name = "assigned_to.cf_#{field.id}"
  1325. assert_include filter_name, query.available_filters.keys
  1326. query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
  1327. assert_equal [3], find_issues_with_query(query).map(&:assigned_to_id).uniq.sort
  1328. end
  1329. def test_filter_on_fixed_version_custom_field
  1330. field = VersionCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
  1331. CustomValue.create!(:custom_field => field, :customized => Version.find(2), :value => 'Foo')
  1332. query = IssueQuery.new(:name => '_')
  1333. filter_name = "fixed_version.cf_#{field.id}"
  1334. assert_include filter_name, query.available_filters.keys
  1335. query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
  1336. assert_equal [2], find_issues_with_query(query).map(&:fixed_version_id).uniq.sort
  1337. end
  1338. def test_filter_on_fixed_version_due_date
  1339. query = IssueQuery.new(:name => '_')
  1340. filter_name = "fixed_version.due_date"
  1341. assert_include filter_name, query.available_filters.keys
  1342. query.filters = {filter_name => {:operator => '=', :values => [20.day.from_now.to_date.to_fs(:db)]}}
  1343. issues = find_issues_with_query(query)
  1344. assert_equal [2], issues.map(&:fixed_version_id).uniq.sort
  1345. assert_equal [2, 12], issues.map(&:id).sort
  1346. query = IssueQuery.new(:name => '_')
  1347. query.filters = {filter_name => {:operator => '>=', :values => [21.day.from_now.to_date.to_fs(:db)]}}
  1348. assert_equal 0, find_issues_with_query(query).size
  1349. end
  1350. def test_filter_on_fixed_version_status
  1351. query = IssueQuery.new(:name => '_')
  1352. filter_name = "fixed_version.status"
  1353. assert_include filter_name, query.available_filters.keys
  1354. query.filters = {filter_name => {:operator => '=', :values => ['closed']}}
  1355. issues = find_issues_with_query(query)
  1356. assert_equal [1], issues.map(&:fixed_version_id).sort
  1357. assert_equal [11], issues.map(&:id).sort
  1358. # "is not" operator should include issues without target version
  1359. query = IssueQuery.new(:name => '_')
  1360. query.filters = {filter_name => {:operator => '!', :values => ['open', 'closed', 'locked']}, "project_id" => {:operator => '=', :values => [1]}}
  1361. assert_equal [1, 3, 7, 8], find_issues_with_query(query).map(&:id).uniq.sort
  1362. end
  1363. def test_filter_on_fixed_version_status_respects_sharing
  1364. issue = Issue.generate!(:project_id => 1, :fixed_version_id => 7)
  1365. filter_name = "fixed_version.status"
  1366. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  1367. assert_include filter_name, query.available_filters.keys
  1368. query.filters = {filter_name => {:operator => '=', :values => ['open']}}
  1369. assert_include issue, find_issues_with_query(query)
  1370. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  1371. query.filters = {filter_name => {:operator => '=', :values => ['closed']}}
  1372. assert_not_includes find_issues_with_query(query), issue
  1373. end
  1374. def test_filter_on_version_custom_field
  1375. field = IssueCustomField.generate!(:field_format => 'version', :is_filter => true)
  1376. issue = Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {field.id.to_s => '2'})
  1377. query = IssueQuery.new(:name => '_')
  1378. filter_name = "cf_#{field.id}"
  1379. assert_include filter_name, query.available_filters.keys
  1380. query.filters = {filter_name => {:operator => '=', :values => ['2']}}
  1381. issues = find_issues_with_query(query)
  1382. assert_equal [issue.id], issues.map(&:id).sort
  1383. end
  1384. def test_filter_on_attribute_of_version_custom_field
  1385. field = IssueCustomField.generate!(:field_format => 'version', :is_filter => true)
  1386. version = Version.generate!(:effective_date => '2017-01-14')
  1387. issue = Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {field.id.to_s => version.id.to_s})
  1388. query = IssueQuery.new(:name => '_')
  1389. filter_name = "cf_#{field.id}.due_date"
  1390. assert_include filter_name, query.available_filters.keys
  1391. query.filters = {filter_name => {:operator => '=', :values => ['2017-01-14']}}
  1392. issues = find_issues_with_query(query)
  1393. assert_equal [issue.id], issues.map(&:id).sort
  1394. end
  1395. def test_filter_on_custom_field_of_version_custom_field
  1396. field = IssueCustomField.generate!(:field_format => 'version', :is_filter => true)
  1397. attr = VersionCustomField.generate!(:field_format => 'string', :is_filter => true)
  1398. version = Version.generate!(:custom_field_values => {attr.id.to_s => 'ABC'})
  1399. issue = Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {field.id.to_s => version.id.to_s})
  1400. query = IssueQuery.new(:name => '_')
  1401. filter_name = "cf_#{field.id}.cf_#{attr.id}"
  1402. assert_include filter_name, query.available_filters.keys
  1403. query.filters = {filter_name => {:operator => '=', :values => ['ABC']}}
  1404. issues = find_issues_with_query(query)
  1405. assert_equal [issue.id], issues.map(&:id).sort
  1406. end
  1407. def test_filter_on_relations_with_a_specific_issue
  1408. IssueRelation.delete_all
  1409. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
  1410. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
  1411. query = IssueQuery.new(:name => '_')
  1412. query.filters = {"relates" => {:operator => '=', :values => ['1']}}
  1413. assert_equal [2, 3], find_issues_with_query(query).map(&:id).sort
  1414. query = IssueQuery.new(:name => '_')
  1415. query.filters = {"relates" => {:operator => '=', :values => ['2']}}
  1416. assert_equal [1], find_issues_with_query(query).map(&:id).sort
  1417. query = IssueQuery.new(:name => '_')
  1418. query.filters = {"relates" => {:operator => '=', :values => ['1,2']}}
  1419. assert_equal [1, 2, 3], find_issues_with_query(query).map(&:id).sort
  1420. query = IssueQuery.new(:name => '_')
  1421. query.filters = {"relates" => {:operator => '=', :values => ['invalid']}}
  1422. assert_equal [], find_issues_with_query(query).map(&:id)
  1423. query = IssueQuery.new(:name => '_')
  1424. query.filters = {"relates" => {:operator => '!', :values => ['1']}}
  1425. assert_equal Issue.where.not(:id => [2, 3]).order(:id).ids, find_issues_with_query(query).map(&:id).sort
  1426. query = IssueQuery.new(:name => '_')
  1427. query.filters = {"relates" => {:operator => '!', :values => ['1,2']}}
  1428. assert_equal Issue.where.not(:id => [1, 2, 3]).order(:id).ids, find_issues_with_query(query).map(&:id).sort
  1429. end
  1430. def test_filter_on_relations_with_any_issues_in_a_project
  1431. IssueRelation.delete_all
  1432. with_settings :cross_project_issue_relations => '1' do
  1433. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
  1434. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(2).issues.first)
  1435. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(3).issues.first)
  1436. end
  1437. query = IssueQuery.new(:name => '_')
  1438. query.filters = {"relates" => {:operator => '=p', :values => ['2']}}
  1439. assert_equal [1, 2], find_issues_with_query(query).map(&:id).sort
  1440. query = IssueQuery.new(:name => '_')
  1441. query.filters = {"relates" => {:operator => '=p', :values => ['3']}}
  1442. assert_equal [1], find_issues_with_query(query).map(&:id).sort
  1443. query = IssueQuery.new(:name => '_')
  1444. query.filters = {"relates" => {:operator => '=p', :values => ['4']}}
  1445. assert_equal [], find_issues_with_query(query).map(&:id).sort
  1446. end
  1447. def test_filter_on_relations_with_any_issues_not_in_a_project
  1448. IssueRelation.delete_all
  1449. with_settings :cross_project_issue_relations => '1' do
  1450. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
  1451. # IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(1).issues.first)
  1452. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(3).issues.first)
  1453. end
  1454. query = IssueQuery.new(:name => '_')
  1455. query.filters = {"relates" => {:operator => '=!p', :values => ['1']}}
  1456. assert_equal [1], find_issues_with_query(query).map(&:id).sort
  1457. end
  1458. def test_filter_on_relations_with_no_issues_in_a_project
  1459. IssueRelation.delete_all
  1460. with_settings :cross_project_issue_relations => '1' do
  1461. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
  1462. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(3).issues.first)
  1463. IssueRelation.create!(:relation_type => "relates", :issue_to => Project.find(2).issues.first, :issue_from => Issue.find(3))
  1464. end
  1465. query = IssueQuery.new(:name => '_')
  1466. query.filters = {"relates" => {:operator => '!p', :values => ['2']}}
  1467. ids = find_issues_with_query(query).map(&:id).sort
  1468. assert_include 2, ids
  1469. assert_not_include 1, ids
  1470. assert_not_include 3, ids
  1471. end
  1472. def test_filter_on_relations_with_any_open_issues
  1473. IssueRelation.delete_all
  1474. # Issue 1 is blocked by 8, which is closed
  1475. IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(1), :issue_to => Issue.find(8))
  1476. # Issue 2 is blocked by 3, which is open
  1477. IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(2), :issue_to => Issue.find(3))
  1478. query = IssueQuery.new(:name => '_')
  1479. query.filters = {"blocked" => {:operator => "*o", :values => ['']}}
  1480. ids = find_issues_with_query(query).map(&:id)
  1481. assert_equal [], ids & [1]
  1482. assert_include 2, ids
  1483. end
  1484. def test_filter_on_blocked_by_no_open_issues
  1485. IssueRelation.delete_all
  1486. # Issue 1 is blocked by 8, which is closed
  1487. IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(1), :issue_to => Issue.find(8))
  1488. # Issue 2 is blocked by 3, which is open
  1489. IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(2), :issue_to => Issue.find(3))
  1490. query = IssueQuery.new(:name => '_')
  1491. query.filters = {"blocked" => {:operator => "!o", :values => ['']}}
  1492. ids = find_issues_with_query(query).map(&:id)
  1493. assert_equal [], ids & [2]
  1494. assert_include 1, ids
  1495. end
  1496. def test_filter_on_related_with_no_open_issues
  1497. IssueRelation.delete_all
  1498. # Issue 1 is blocked by 8, which is closed
  1499. IssueRelation.create!(relation_type: 'relates', issue_from: Issue.find(1), issue_to: Issue.find(8))
  1500. # Issue 2 is blocked by 3, which is open
  1501. IssueRelation.create!(relation_type: 'relates', issue_from: Issue.find(2), issue_to: Issue.find(3))
  1502. query = IssueQuery.new(:name => '_')
  1503. query.filters = {'relates' => {:operator => '!o', :values => ['']}}
  1504. ids = find_issues_with_query(query).map(&:id)
  1505. assert_equal [], ids & [2]
  1506. assert_include 1, ids
  1507. end
  1508. def test_filter_on_relations_with_no_issues
  1509. IssueRelation.delete_all
  1510. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
  1511. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
  1512. query = IssueQuery.new(:name => '_')
  1513. query.filters = {"relates" => {:operator => '!*', :values => ['']}}
  1514. ids = find_issues_with_query(query).map(&:id)
  1515. assert_equal [], ids & [1, 2, 3]
  1516. assert_include 4, ids
  1517. end
  1518. def test_filter_on_relations_with_any_issues
  1519. IssueRelation.delete_all
  1520. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
  1521. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
  1522. query = IssueQuery.new(:name => '_')
  1523. query.filters = {"relates" => {:operator => '*', :values => ['']}}
  1524. assert_equal [1, 2, 3], find_issues_with_query(query).map(&:id).sort
  1525. end
  1526. def test_filter_on_relations_should_not_ignore_other_filter
  1527. issue = Issue.generate!
  1528. issue1 = Issue.generate!(:status_id => 1)
  1529. issue2 = Issue.generate!(:status_id => 2)
  1530. IssueRelation.create!(:relation_type => "relates", :issue_from => issue, :issue_to => issue1)
  1531. IssueRelation.create!(:relation_type => "relates", :issue_from => issue, :issue_to => issue2)
  1532. query = IssueQuery.new(:name => '_')
  1533. query.filters = {
  1534. "status_id" => {:operator => '=', :values => ['1']},
  1535. "relates" => {:operator => '=', :values => [issue.id.to_s]}
  1536. }
  1537. assert_equal [issue1], find_issues_with_query(query)
  1538. end
  1539. def test_filter_on_parent
  1540. Issue.delete_all
  1541. parent = Issue.generate_with_descendants!
  1542. query = IssueQuery.new(:name => '_')
  1543. query.filters = {"parent_id" => {:operator => '=', :values => [parent.id.to_s]}}
  1544. assert_equal parent.children.map(&:id).sort, find_issues_with_query(query).map(&:id).sort
  1545. query.filters = {"parent_id" => {:operator => '~', :values => [parent.id.to_s]}}
  1546. assert_equal parent.descendants.map(&:id).sort, find_issues_with_query(query).map(&:id).sort
  1547. query.filters = {"parent_id" => {:operator => '*', :values => ['']}}
  1548. assert_equal parent.descendants.map(&:id).sort, find_issues_with_query(query).map(&:id).sort
  1549. query.filters = {"parent_id" => {:operator => '!*', :values => ['']}}
  1550. assert_equal [parent.id], find_issues_with_query(query).map(&:id).sort
  1551. end
  1552. def test_filter_on_invalid_parent_should_return_no_results
  1553. query = IssueQuery.new(:name => '_')
  1554. query.filters = {"parent_id" => {:operator => '=', :values => '99999999999'}}
  1555. assert_equal [], find_issues_with_query(query).map(&:id).sort
  1556. query.filters = {"parent_id" => {:operator => '~', :values => '99999999999'}}
  1557. assert_equal [], find_issues_with_query(query)
  1558. end
  1559. def test_operator_contains_on_parent_id_should_accept_comma_separated_values
  1560. parent1 = Issue.generate!
  1561. children_of_parent1 = [
  1562. Issue.generate!(parent_id: parent1.id),
  1563. Issue.generate!(parent_id: parent1.id)
  1564. ]
  1565. parent2 = Issue.generate!
  1566. children_of_parent2 = [
  1567. Issue.generate!(parent_id: parent2.id),
  1568. Issue.generate!(parent_id: parent2.id)
  1569. ]
  1570. grandchild_of_parent2 = [
  1571. Issue.generate!(parent_id: children_of_parent2.first.id)
  1572. ]
  1573. query = IssueQuery.new(name: '_')
  1574. query.add_filter('parent_id', '~', ["#{parent1.id},#{parent2.id}"])
  1575. issues = find_issues_with_query(query)
  1576. expected =
  1577. children_of_parent1 + children_of_parent2 + grandchild_of_parent2
  1578. assert_equal expected.size, issues.size
  1579. assert_equal expected.map(&:id).sort, issues.map(&:id).sort
  1580. end
  1581. def test_filter_on_child
  1582. Issue.delete_all
  1583. parent = Issue.generate_with_descendants!
  1584. child, leaf = parent.children.sort_by(&:id)
  1585. grandchild = child.children.first
  1586. query = IssueQuery.new(:name => '_')
  1587. query.filters = {"child_id" => {:operator => '=', :values => [grandchild.id.to_s]}}
  1588. assert_equal [child.id], find_issues_with_query(query).map(&:id).sort
  1589. query.filters = {"child_id" => {:operator => '~', :values => [grandchild.id.to_s]}}
  1590. assert_equal [parent, child].map(&:id).sort, find_issues_with_query(query).map(&:id).sort
  1591. query.filters = {"child_id" => {:operator => '*', :values => ['']}}
  1592. assert_equal [parent, child].map(&:id).sort, find_issues_with_query(query).map(&:id).sort
  1593. query.filters = {"child_id" => {:operator => '!*', :values => ['']}}
  1594. assert_equal [grandchild, leaf].map(&:id).sort, find_issues_with_query(query).map(&:id).sort
  1595. end
  1596. def test_filter_on_invalid_child_should_return_no_results
  1597. query = IssueQuery.new(:name => '_')
  1598. query.filters = {"child_id" => {:operator => '=', :values => '99999999999'}}
  1599. assert_equal [], find_issues_with_query(query)
  1600. query.filters = {"child_id" => {:operator => '~', :values => '99999999999'}}
  1601. assert_equal [].map(&:id).sort, find_issues_with_query(query)
  1602. end
  1603. def test_filter_on_attachment_any
  1604. query = IssueQuery.new(:name => '_')
  1605. query.filters = {"attachment" => {:operator => '*', :values => ['']}}
  1606. issues = find_issues_with_query(query)
  1607. assert issues.any?
  1608. assert_nil issues.detect {|issue| issue.attachments.empty?}
  1609. end
  1610. def test_filter_on_attachment_none
  1611. query = IssueQuery.new(:name => '_')
  1612. query.filters = {"attachment" => {:operator => '!*', :values => ['']}}
  1613. issues = find_issues_with_query(query)
  1614. assert issues.any?
  1615. assert_nil issues.detect {|issue| issue.attachments.any?}
  1616. end
  1617. def test_filter_on_attachment_contains
  1618. query = IssueQuery.new(:name => '_')
  1619. query.filters = {"attachment" => {:operator => '~', :values => ['error281']}}
  1620. issues = find_issues_with_query(query)
  1621. assert issues.any?
  1622. assert_nil issues.detect {|issue| ! issue.attachments.any? {|attachment| attachment.filename.include?('error281')}}
  1623. end
  1624. def test_filter_on_attachment_not_contains
  1625. query = IssueQuery.new(:name => '_')
  1626. query.filters = {"attachment" => {:operator => '!~', :values => ['error281']}}
  1627. issues = find_issues_with_query(query)
  1628. assert issues.any?
  1629. assert_nil issues.detect {|issue| issue.attachments.any? {|attachment| attachment.filename.include?('error281')}}
  1630. end
  1631. def test_filter_on_attachment_when_starts_with
  1632. query = IssueQuery.new(:name => '_')
  1633. query.filters = {"attachment" => {:operator => '^', :values => ['testfile']}}
  1634. issues = find_issues_with_query(query)
  1635. assert_equal [14], issues.collect(&:id).sort
  1636. end
  1637. def test_filter_on_attachment_when_ends_with
  1638. query = IssueQuery.new(:name => '_')
  1639. query.filters = {"attachment" => {:operator => '$', :values => ['zip']}}
  1640. issues = find_issues_with_query(query)
  1641. assert_equal [3, 4], issues.collect(&:id).sort
  1642. end
  1643. def test_filter_on_attachment_description_when_any
  1644. query = IssueQuery.new(:name => '_')
  1645. query.filters = {"attachment_description" => {:operator => '*', :values => ['']}}
  1646. issues = find_issues_with_query(query)
  1647. assert_equal [2, 3, 14], issues.collect(&:id).sort
  1648. end
  1649. def test_filter_on_attachment_description_when_none
  1650. query = IssueQuery.new(:name => '_')
  1651. query.filters = {"attachment_description" => {:operator => '!*', :values => ['']}}
  1652. issues = find_issues_with_query(query)
  1653. assert_equal [2, 3, 4, 14], issues.collect(&:id).sort
  1654. end
  1655. def test_filter_on_attachment_description_when_contains
  1656. query = IssueQuery.new(:name => '_')
  1657. query.filters = {"attachment_description" => {:operator => '~', :values => ['attachment']}}
  1658. issues = find_issues_with_query(query)
  1659. assert_equal [3, 14], issues.collect(&:id).sort
  1660. end
  1661. def test_filter_on_attachment_description_when_does_not_contain
  1662. query = IssueQuery.new(:name => '_')
  1663. query.filters = {"attachment_description" => {:operator => '!~', :values => ['attachment']}}
  1664. issues = find_issues_with_query(query)
  1665. assert_equal [2], issues.collect(&:id).sort
  1666. end
  1667. def test_filter_on_attachment_description_when_starts_with
  1668. query = IssueQuery.new(:name => '_')
  1669. query.filters = {"attachment_description" => {:operator => '^', :values => ['attachment']}}
  1670. issues = find_issues_with_query(query)
  1671. assert_equal [14], issues.collect(&:id).sort
  1672. end
  1673. def test_filter_on_attachment_description_when_ends_with
  1674. query = IssueQuery.new(:name => '_')
  1675. query.filters = {"attachment_description" => {:operator => '$', :values => ['attachment']}}
  1676. issues = find_issues_with_query(query)
  1677. assert_equal [3], issues.collect(&:id).sort
  1678. end
  1679. def test_filter_on_subject_when_starts_with
  1680. query = IssueQuery.new(:name => '_')
  1681. query.filters = {'subject' => {:operator => '^', :values => ['issue']}}
  1682. issues = find_issues_with_query(query)
  1683. assert_equal [4, 6, 7, 10], issues.collect(&:id).sort
  1684. end
  1685. def test_filter_on_subject_when_ends_with
  1686. query = IssueQuery.new(:name => '_')
  1687. query.filters = {'subject' => {:operator => '$', :values => ['issue']}}
  1688. issues = find_issues_with_query(query)
  1689. assert_equal [5, 8, 9], issues.collect(&:id).sort
  1690. end
  1691. def test_statement_should_be_nil_with_no_filters
  1692. q = IssueQuery.new(:name => '_')
  1693. q.filters = {}
  1694. assert q.valid?
  1695. assert_nil q.statement
  1696. end
  1697. def test_available_filters_as_json_should_include_missing_assigned_to_id_values
  1698. user = User.generate!
  1699. with_current_user User.find(1) do
  1700. q = IssueQuery.new
  1701. q.filters = {"assigned_to_id" => {:operator => '=', :values => user.id.to_s}}
  1702. filters = q.available_filters_as_json
  1703. assert_include [user.name, user.id.to_s], filters['assigned_to_id']['values']
  1704. end
  1705. end
  1706. def test_available_filters_as_json_should_not_include_duplicate_assigned_to_id_values
  1707. set_language_if_valid 'en'
  1708. user = User.find_by_login 'dlopper'
  1709. with_current_user User.find(1) do
  1710. q = IssueQuery.new
  1711. q.filters = {"assigned_to_id" => {:operator => '=', :values => user.id.to_s}}
  1712. filters = q.available_filters_as_json
  1713. assert_not_include [user.name, user.id.to_s], filters['assigned_to_id']['values']
  1714. assert_include [user.name, user.id.to_s, 'active'], filters['assigned_to_id']['values']
  1715. end
  1716. end
  1717. def test_available_filters_as_json_should_include_missing_author_id_values
  1718. user = User.generate!
  1719. with_current_user User.find(1) do
  1720. q = IssueQuery.new
  1721. q.filters = {"author_id" => {:operator => '=', :values => user.id.to_s}}
  1722. filters = q.available_filters_as_json
  1723. assert_include [user.name, user.id.to_s], filters['author_id']['values']
  1724. end
  1725. end
  1726. def test_default_columns
  1727. q = IssueQuery.new
  1728. assert q.columns.any?
  1729. assert q.inline_columns.any?
  1730. assert q.block_columns.empty?
  1731. end
  1732. def test_set_column_names
  1733. q = IssueQuery.new
  1734. q.column_names = ['tracker', :subject, '', 'unknonw_column']
  1735. assert_equal [:id, :tracker, :subject], q.columns.collect {|c| c.name}
  1736. end
  1737. def test_has_column_should_accept_a_column_name
  1738. q = IssueQuery.new
  1739. q.column_names = ['tracker', :subject]
  1740. assert q.has_column?(:tracker)
  1741. assert !q.has_column?(:category)
  1742. end
  1743. def test_has_column_should_accept_a_column
  1744. q = IssueQuery.new
  1745. q.column_names = ['tracker', :subject]
  1746. tracker_column = q.available_columns.detect {|c| c.name==:tracker}
  1747. assert_kind_of QueryColumn, tracker_column
  1748. category_column = q.available_columns.detect {|c| c.name==:category}
  1749. assert_kind_of QueryColumn, category_column
  1750. assert q.has_column?(tracker_column)
  1751. assert !q.has_column?(category_column)
  1752. end
  1753. def test_has_column_should_return_true_for_default_column
  1754. with_settings :issue_list_default_columns => %w(tracker subject) do
  1755. q = IssueQuery.new
  1756. assert q.has_column?(:tracker)
  1757. assert !q.has_column?(:category)
  1758. end
  1759. end
  1760. def test_inline_and_block_columns
  1761. q = IssueQuery.new
  1762. q.column_names = ['subject', 'description', 'tracker', 'last_notes']
  1763. assert_equal [:id, :subject, :tracker], q.inline_columns.map(&:name)
  1764. assert_equal [:description, :last_notes], q.block_columns.map(&:name)
  1765. end
  1766. def test_custom_field_columns_should_be_inline
  1767. q = IssueQuery.new
  1768. columns = q.available_columns.select {|column| column.is_a? QueryCustomFieldColumn}
  1769. assert columns.any?
  1770. assert_nil columns.detect {|column| !column.inline?}
  1771. end
  1772. def test_query_should_preload_spent_hours
  1773. q = IssueQuery.new(:name => '_', :column_names => [:subject, :spent_hours])
  1774. assert q.has_column?(:spent_hours)
  1775. issues = q.issues
  1776. assert_not_nil issues.first.instance_variable_get(:@spent_hours)
  1777. end
  1778. def test_query_should_preload_last_updated_by
  1779. with_current_user User.find(2) do
  1780. q = IssueQuery.new(:name => '_', :column_names => [:subject, :last_updated_by])
  1781. q.filters = {"issue_id" => {:operator => '=', :values => ['1,2,3']}}
  1782. assert q.has_column?(:last_updated_by)
  1783. issues = q.issues.sort_by(&:id)
  1784. assert issues.all? {|issue| !issue.instance_variable_get(:@last_updated_by).nil?}
  1785. assert_equal ["User", "User", "NilClass"], issues.map {|i| i.last_updated_by.class.name}
  1786. assert_equal ["John Smith", "John Smith", ""], issues.map {|i| i.last_updated_by.to_s}
  1787. end
  1788. end
  1789. def test_query_should_preload_last_notes
  1790. q = IssueQuery.new(:name => '_', :column_names => [:subject, :last_notes])
  1791. assert q.has_column?(:last_notes)
  1792. issues = q.issues
  1793. assert_not_nil issues.first.instance_variable_get(:@last_notes)
  1794. end
  1795. def test_groupable_columns_should_include_custom_fields
  1796. q = IssueQuery.new
  1797. column = q.groupable_columns.detect {|c| c.name == :cf_1}
  1798. assert_not_nil column
  1799. assert_kind_of QueryCustomFieldColumn, column
  1800. end
  1801. def test_groupable_columns_should_not_include_multi_custom_fields
  1802. field = CustomField.find(1)
  1803. field.update_attribute :multiple, true
  1804. q = IssueQuery.new
  1805. column = q.groupable_columns.detect {|c| c.name == :cf_1}
  1806. assert_nil column
  1807. end
  1808. def test_groupable_columns_should_include_user_custom_fields
  1809. cf =
  1810. IssueCustomField.create!(
  1811. :name => 'User', :is_for_all => true, :tracker_ids => [1],
  1812. :field_format => 'user'
  1813. )
  1814. q = IssueQuery.new
  1815. assert q.groupable_columns.detect {|c| c.name == :"cf_#{cf.id}"}
  1816. end
  1817. def test_groupable_columns_should_include_version_custom_fields
  1818. cf =
  1819. IssueCustomField.create!(
  1820. :name => 'User', :is_for_all => true,
  1821. :tracker_ids => [1], :field_format => 'version'
  1822. )
  1823. q = IssueQuery.new
  1824. assert q.groupable_columns.detect {|c| c.name == :"cf_#{cf.id}"}
  1825. end
  1826. def test_grouped_with_valid_column
  1827. q = IssueQuery.new(:group_by => 'status')
  1828. assert q.grouped?
  1829. assert_not_nil q.group_by_column
  1830. assert_equal :status, q.group_by_column.name
  1831. assert_not_nil q.group_by_statement
  1832. assert_equal 'status', q.group_by_statement
  1833. end
  1834. def test_grouped_with_invalid_column
  1835. q = IssueQuery.new(:group_by => 'foo')
  1836. assert !q.grouped?
  1837. assert_nil q.group_by_column
  1838. assert_nil q.group_by_statement
  1839. end
  1840. def test_sortable_columns_should_sort_assignees_according_to_user_format_setting
  1841. with_settings :user_format => 'lastname_comma_firstname' do
  1842. q = IssueQuery.new
  1843. assert q.sortable_columns.has_key?('assigned_to')
  1844. assert_equal %w(users.lastname users.firstname users.id), q.sortable_columns['assigned_to']
  1845. end
  1846. end
  1847. def test_sortable_columns_should_sort_authors_according_to_user_format_setting
  1848. with_settings :user_format => 'lastname_comma_firstname' do
  1849. q = IssueQuery.new
  1850. assert q.sortable_columns.has_key?('author')
  1851. assert_equal %w(authors.lastname authors.firstname authors.id), q.sortable_columns['author']
  1852. end
  1853. end
  1854. def test_sortable_columns_should_sort_last_updated_by_according_to_user_format_setting
  1855. with_settings :user_format => 'lastname_comma_firstname' do
  1856. q = IssueQuery.new
  1857. q.sort_criteria = [['last_updated_by', 'desc']]
  1858. assert q.sortable_columns.has_key?('last_updated_by')
  1859. assert_equal(
  1860. %w(last_journal_user.lastname last_journal_user.firstname last_journal_user.id),
  1861. q.sortable_columns['last_updated_by']
  1862. )
  1863. end
  1864. end
  1865. def test_sortable_columns_should_include_custom_field
  1866. q = IssueQuery.new
  1867. assert q.sortable_columns['cf_1']
  1868. end
  1869. def test_sortable_columns_should_not_include_multi_custom_field
  1870. field = CustomField.find(1)
  1871. field.update_attribute :multiple, true
  1872. q = IssueQuery.new
  1873. assert !q.sortable_columns['cf_1']
  1874. end
  1875. def test_sortable_should_return_false_for_multi_custom_field
  1876. field = CustomField.find(1)
  1877. field.update_attribute :multiple, true
  1878. q = IssueQuery.new
  1879. field_column = q.available_columns.detect {|c| c.name==:cf_1}
  1880. assert !field_column.sortable?
  1881. end
  1882. def test_default_sort
  1883. q = IssueQuery.new
  1884. assert_equal [['id', 'desc']], q.sort_criteria
  1885. end
  1886. def test_sort_criteria_should_have_only_first_three_elements
  1887. q = IssueQuery.new
  1888. q.sort_criteria = [
  1889. ['priority', 'desc'], ['tracker', 'asc'], ['priority', 'asc'],
  1890. ['id', 'asc'], ['project', 'asc'], ['subject', 'asc']
  1891. ]
  1892. assert_equal [['priority', 'desc'], ['tracker', 'asc'], ['id', 'asc']], q.sort_criteria
  1893. end
  1894. def test_sort_criteria_should_remove_blank_or_duplicate_keys
  1895. q = IssueQuery.new
  1896. q.sort_criteria = [['priority', 'desc'], [nil, 'desc'], ['', 'asc'], ['priority', 'asc'], ['project', 'asc']]
  1897. assert_equal [['priority', 'desc'], ['project', 'asc']], q.sort_criteria
  1898. end
  1899. def test_set_sort_criteria_with_hash
  1900. q = IssueQuery.new
  1901. q.sort_criteria = {'0' => ['priority', 'desc'], '2' => ['tracker']}
  1902. assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
  1903. end
  1904. def test_set_sort_criteria_with_array
  1905. q = IssueQuery.new
  1906. q.sort_criteria = [['priority', 'desc'], 'tracker']
  1907. assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
  1908. end
  1909. def test_create_query_with_sort
  1910. q = IssueQuery.new(:name => 'Sorted')
  1911. q.sort_criteria = [['priority', 'desc'], 'tracker']
  1912. assert q.save
  1913. q.reload
  1914. assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
  1915. end
  1916. def test_sort_by_string_custom_field_asc
  1917. q = IssueQuery.new
  1918. c =
  1919. q.available_columns.find do |col|
  1920. col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'string'
  1921. end
  1922. assert c
  1923. assert c.sortable
  1924. q.sort_criteria = [[c.name.to_s, 'asc']]
  1925. issues = q.issues
  1926. values = issues.collect {|i| i.custom_value_for(c.custom_field).to_s}
  1927. assert !values.empty?
  1928. assert_equal values.sort, values
  1929. end
  1930. def test_sort_by_string_custom_field_desc
  1931. q = IssueQuery.new
  1932. c =
  1933. q.available_columns.find do |col|
  1934. col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'string'
  1935. end
  1936. assert c
  1937. assert c.sortable
  1938. q.sort_criteria = [[c.name.to_s, 'desc']]
  1939. issues = q.issues
  1940. values = issues.collect {|i| i.custom_value_for(c.custom_field).to_s}
  1941. assert !values.empty?
  1942. assert_equal values.sort.reverse, values
  1943. end
  1944. def test_sort_by_float_custom_field_asc
  1945. q = IssueQuery.new
  1946. c =
  1947. q.available_columns.find do |col|
  1948. col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'float'
  1949. end
  1950. assert c
  1951. assert c.sortable
  1952. q.sort_criteria = [[c.name.to_s, 'asc']]
  1953. issues = q.issues
  1954. values =
  1955. issues.filter_map do |i|
  1956. begin
  1957. Kernel.Float(i.custom_value_for(c.custom_field).to_s)
  1958. rescue
  1959. nil
  1960. end
  1961. end
  1962. assert !values.empty?
  1963. assert_equal values.sort, values
  1964. end
  1965. def test_sort_with_group_by_timestamp_query_column_should_sort_after_date_value
  1966. User.current = User.find(1)
  1967. # Touch Issue#10 in order to be the last updated issue
  1968. Issue.find(10).update_attribute(:updated_on, Issue.find(10).updated_on + 1)
  1969. q = IssueQuery.new(
  1970. :name => '_',
  1971. :filters => {'updated_on' => {:operator => 't', :values => ['']}},
  1972. :group_by => 'updated_on',
  1973. :sort_criteria => [['subject', 'asc']]
  1974. )
  1975. # The following 3 issues are updated today (ordered by updated_on):
  1976. # Issue#10: Issue Doing the Blocking
  1977. # Issue#9: Blocked Issue
  1978. # Issue#6: Issue of a private subproject
  1979. # When we group by a timestamp query column, all the issues in the group have the same date value (today)
  1980. # and the time of the value should not be taken into consideration when sorting
  1981. #
  1982. # For the same issues after subject ascending should return the following:
  1983. # Issue#9: Blocked Issue
  1984. # Issue#10: Issue Doing the Blocking
  1985. # Issue#6: Issue of a private subproject
  1986. assert_equal [9, 10, 6], q.issues.map(&:id)
  1987. end
  1988. def test_sort_by_total_for_estimated_hours
  1989. # Prepare issues
  1990. parent = issues(:issues_001)
  1991. child = issues(:issues_002)
  1992. private_child = issues(:issues_003)
  1993. other = issues(:issues_007)
  1994. User.current = users(:users_001)
  1995. parent.safe_attributes = {:estimated_hours => 1}
  1996. child.safe_attributes = {:estimated_hours => 2, :parent_issue_id => 1}
  1997. private_child.safe_attributes = {:estimated_hours => 4, :parent_issue_id => 1, :is_private => true}
  1998. other.safe_attributes = {:estimated_hours => 5}
  1999. [parent, child, private_child, other].each(&:save!)
  2000. q = IssueQuery.new(
  2001. :name => '_',
  2002. :filters => {'issue_id' => {:operator => '=', :values => ['1,7']}},
  2003. :sort_criteria => [['total_estimated_hours', 'asc']]
  2004. )
  2005. # With private_child, `parent' is "bigger" than `other'
  2006. ids = q.issue_ids
  2007. assert_equal [7, 1], ids, "Private issue was not used to calculate sort order"
  2008. # Without the invisible private_child, `other' is "bigger" than `parent'
  2009. User.current = User.anonymous
  2010. ids = q.issue_ids
  2011. assert_equal [1, 7], ids, "Private issue was used to calculate sort order"
  2012. end
  2013. def test_set_totalable_names
  2014. q = IssueQuery.new
  2015. q.totalable_names = ['estimated_hours', :spent_hours, '']
  2016. assert_equal [:estimated_hours, :spent_hours], q.totalable_columns.map(&:name)
  2017. end
  2018. def test_totalable_columns_should_default_to_settings
  2019. with_settings :issue_list_default_totals => ['estimated_hours'] do
  2020. q = IssueQuery.new
  2021. assert_equal [:estimated_hours], q.totalable_columns.map(&:name)
  2022. end
  2023. end
  2024. def test_available_totalable_columns_should_include_estimated_hours
  2025. q = IssueQuery.new
  2026. assert_include :estimated_hours, q.available_totalable_columns.map(&:name)
  2027. end
  2028. def test_available_totalable_columns_should_include_estimated_remaining_hours
  2029. q = IssueQuery.new
  2030. assert_include :estimated_remaining_hours, q.available_totalable_columns.map(&:name)
  2031. end
  2032. def test_available_totalable_columns_should_include_spent_hours
  2033. User.current = User.find(1)
  2034. q = IssueQuery.new
  2035. assert_include :spent_hours, q.available_totalable_columns.map(&:name)
  2036. end
  2037. def test_available_totalable_columns_should_include_int_custom_field
  2038. field = IssueCustomField.generate!(:field_format => 'int', :is_for_all => true)
  2039. q = IssueQuery.new
  2040. assert_include :"cf_#{field.id}", q.available_totalable_columns.map(&:name)
  2041. end
  2042. def test_available_totalable_columns_should_include_float_custom_field
  2043. field = IssueCustomField.generate!(:field_format => 'float', :is_for_all => true)
  2044. q = IssueQuery.new
  2045. assert_include :"cf_#{field.id}", q.available_totalable_columns.map(&:name)
  2046. end
  2047. def test_available_totalable_columns_should_sort_in_position_order_for_custom_field
  2048. ProjectCustomField.delete_all
  2049. cf_pos3 = ProjectCustomField.generate!(:position => 3, :is_for_all => true, :field_format => 'int')
  2050. cf_pos4 = ProjectCustomField.generate!(:position => 4, :is_for_all => true, :field_format => 'float')
  2051. cf_pos1 = ProjectCustomField.generate!(:position => 1, :is_for_all => true, :field_format => 'float')
  2052. cf_pos2 = ProjectCustomField.generate!(:position => 2, :is_for_all => true, :field_format => 'int')
  2053. q = ProjectQuery.new
  2054. custom_field_columns = q.available_totalable_columns.select{|column| column.is_a?(QueryCustomFieldColumn)}
  2055. assert_equal [cf_pos1, cf_pos2, cf_pos3, cf_pos4], custom_field_columns.collect(&:custom_field)
  2056. IssueCustomField.delete_all
  2057. cf_pos3 = IssueCustomField.generate!(:position => 3, :is_for_all => true, :field_format => 'int')
  2058. cf_pos4 = IssueCustomField.generate!(:position => 4, :is_for_all => true, :field_format => 'float')
  2059. cf_pos1 = IssueCustomField.generate!(:position => 1, :is_for_all => true, :field_format => 'float')
  2060. cf_pos2 = IssueCustomField.generate!(:position => 2, :is_for_all => true, :field_format => 'int')
  2061. q = IssueQuery.new
  2062. custom_field_columns = q.available_totalable_columns.select{|column| column.is_a?(QueryCustomFieldColumn)}
  2063. assert_equal [cf_pos1, cf_pos2, cf_pos3, cf_pos4], custom_field_columns.collect(&:custom_field)
  2064. ProjectCustomField.delete_all
  2065. IssueCustomField.delete_all
  2066. TimeEntryCustomField.delete_all
  2067. cf_pos3 = TimeEntryCustomField.generate!(:position => 3, :is_for_all => true, :field_format => 'int')
  2068. cf_pos4 = TimeEntryCustomField.generate!(:position => 4, :is_for_all => true, :field_format => 'float')
  2069. cf_pos1 = TimeEntryCustomField.generate!(:position => 1, :is_for_all => true, :field_format => 'float')
  2070. cf_pos2 = TimeEntryCustomField.generate!(:position => 2, :is_for_all => true, :field_format => 'int')
  2071. q = TimeEntryQuery.new
  2072. custom_field_columns = q.available_totalable_columns.select{|column| column.is_a?(QueryCustomFieldColumn)}
  2073. assert_equal [cf_pos1, cf_pos2, cf_pos3, cf_pos4], custom_field_columns.collect(&:custom_field)
  2074. end
  2075. def test_total_for_estimated_hours
  2076. Issue.delete_all
  2077. Issue.generate!(:estimated_hours => 5.5)
  2078. Issue.generate!(:estimated_hours => 1.1)
  2079. Issue.generate!
  2080. q = IssueQuery.new
  2081. assert_equal 6.6, q.total_for(:estimated_hours)
  2082. end
  2083. def test_total_by_group_for_estimated_hours
  2084. Issue.delete_all
  2085. Issue.generate!(:estimated_hours => 5.5, :assigned_to_id => 2)
  2086. Issue.generate!(:estimated_hours => 1.1, :assigned_to_id => 3)
  2087. Issue.generate!(:estimated_hours => 3.5)
  2088. q = IssueQuery.new(:group_by => 'assigned_to')
  2089. assert_equal(
  2090. {nil => 3.5, User.find(2) => 5.5, User.find(3) => 1.1},
  2091. q.total_by_group_for(:estimated_hours)
  2092. )
  2093. end
  2094. def test_total_for_estimated_remaining_hours
  2095. Issue.delete_all
  2096. Issue.generate!(:estimated_hours => 5.5, :done_ratio => 50)
  2097. Issue.generate!(:estimated_hours => 1.1, :done_ratio => 100)
  2098. Issue.generate!
  2099. q = IssueQuery.new
  2100. assert_equal 2.75, q.total_for(:estimated_remaining_hours)
  2101. end
  2102. def test_total_by_group_for_estimated_remaining_hours
  2103. Issue.delete_all
  2104. Issue.generate!(:estimated_hours => 5.5, :assigned_to_id => 2, :done_ratio => 50)
  2105. Issue.generate!(:estimated_hours => 1.1, :assigned_to_id => 3, :done_ratio => 100)
  2106. Issue.generate!(:estimated_hours => 3.5, :done_ratio => 0)
  2107. q = IssueQuery.new(:group_by => 'assigned_to')
  2108. assert_equal(
  2109. {nil => 3.5, User.find(2) => 2.75, User.find(3) => 0},
  2110. q.total_by_group_for(:estimated_remaining_hours)
  2111. )
  2112. end
  2113. def test_total_for_spent_hours
  2114. TimeEntry.delete_all
  2115. TimeEntry.generate!(:hours => 5.5)
  2116. TimeEntry.generate!(:hours => 1.1)
  2117. q = IssueQuery.new
  2118. assert_equal 6.6, q.total_for(:spent_hours)
  2119. end
  2120. def test_total_by_group_for_spent_hours
  2121. TimeEntry.delete_all
  2122. TimeEntry.generate!(:hours => 5.5, :issue_id => 1)
  2123. TimeEntry.generate!(:hours => 1.1, :issue_id => 2)
  2124. Issue.where(:id => 1).update_all(:assigned_to_id => 2)
  2125. Issue.where(:id => 2).update_all(:assigned_to_id => 3)
  2126. q = IssueQuery.new(:group_by => 'assigned_to')
  2127. assert_equal(
  2128. {User.find(2) => 5.5, User.find(3) => 1.1},
  2129. q.total_by_group_for(:spent_hours)
  2130. )
  2131. end
  2132. def test_total_by_project_group_for_spent_hours
  2133. TimeEntry.delete_all
  2134. TimeEntry.generate!(:hours => 5.5, :issue_id => 1)
  2135. TimeEntry.generate!(:hours => 1.1, :issue_id => 2)
  2136. Issue.where(:id => 1).update_all(:assigned_to_id => 2)
  2137. Issue.where(:id => 2).update_all(:assigned_to_id => 3)
  2138. q = IssueQuery.new(:group_by => 'project')
  2139. assert_equal(
  2140. {Project.find(1) => 6.6},
  2141. q.total_by_group_for(:spent_hours)
  2142. )
  2143. end
  2144. def test_total_for_int_custom_field
  2145. field = IssueCustomField.generate!(:field_format => 'int', :is_for_all => true)
  2146. CustomValue.create!(:customized => Issue.find(1), :custom_field => field, :value => '2')
  2147. CustomValue.create!(:customized => Issue.find(2), :custom_field => field, :value => '7')
  2148. CustomValue.create!(:customized => Issue.find(3), :custom_field => field, :value => '')
  2149. q = IssueQuery.new
  2150. assert_equal 9, q.total_for("cf_#{field.id}")
  2151. end
  2152. def test_total_by_group_for_int_custom_field
  2153. field = IssueCustomField.generate!(:field_format => 'int', :is_for_all => true)
  2154. CustomValue.create!(:customized => Issue.find(1), :custom_field => field, :value => '2')
  2155. CustomValue.create!(:customized => Issue.find(2), :custom_field => field, :value => '7')
  2156. Issue.where(:id => 1).update_all(:assigned_to_id => 2)
  2157. Issue.where(:id => 2).update_all(:assigned_to_id => 3)
  2158. q = IssueQuery.new(:group_by => 'assigned_to')
  2159. assert_equal(
  2160. {User.find(2) => 2, User.find(3) => 7},
  2161. q.total_by_group_for("cf_#{field.id}")
  2162. )
  2163. end
  2164. def test_total_for_float_custom_field
  2165. field = IssueCustomField.generate!(:field_format => 'float', :is_for_all => true)
  2166. CustomValue.create!(:customized => Issue.find(1), :custom_field => field, :value => '2.3')
  2167. CustomValue.create!(:customized => Issue.find(2), :custom_field => field, :value => '7')
  2168. CustomValue.create!(:customized => Issue.find(3), :custom_field => field, :value => '')
  2169. q = IssueQuery.new
  2170. assert_equal 9.3, q.total_for("cf_#{field.id}")
  2171. end
  2172. def test_invalid_query_should_raise_query_statement_invalid_error
  2173. q = IssueQuery.new
  2174. assert_raise Query::StatementInvalid do
  2175. q.issues(:conditions => "foo = 1")
  2176. end
  2177. end
  2178. def test_issue_count
  2179. q = IssueQuery.new(:name => '_')
  2180. issue_count = q.issue_count
  2181. assert_equal q.issues.size, issue_count
  2182. end
  2183. def test_issue_count_with_archived_issues
  2184. p = Project.generate! do |project|
  2185. project.status = Project::STATUS_ARCHIVED
  2186. end
  2187. i = Issue.generate!(:project => p, :tracker => p.trackers.first)
  2188. assert !i.visible?
  2189. test_issue_count
  2190. end
  2191. def test_issue_count_by_association_group
  2192. q = IssueQuery.new(:name => '_', :group_by => 'assigned_to')
  2193. count_by_group = q.result_count_by_group
  2194. assert_kind_of Hash, count_by_group
  2195. assert_equal %w(NilClass User), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
  2196. assert_equal %w(Integer), count_by_group.values.collect {|k| k.class.name}.uniq
  2197. assert count_by_group.has_key?(User.find(3))
  2198. end
  2199. def test_issue_count_by_list_custom_field_group
  2200. q = IssueQuery.new(:name => '_', :group_by => 'cf_1')
  2201. count_by_group = q.result_count_by_group
  2202. assert_kind_of Hash, count_by_group
  2203. assert_equal %w(NilClass String), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
  2204. assert_equal %w(Integer), count_by_group.values.collect {|k| k.class.name}.uniq
  2205. assert count_by_group.has_key?('MySQL')
  2206. end
  2207. def test_issue_count_by_date_custom_field_group
  2208. q = IssueQuery.new(:name => '_', :group_by => 'cf_8')
  2209. count_by_group = q.result_count_by_group
  2210. assert_kind_of Hash, count_by_group
  2211. assert_equal %w(Date NilClass), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
  2212. assert_equal %w(Integer), count_by_group.values.collect {|k| k.class.name}.uniq
  2213. end
  2214. def test_issue_count_with_nil_group_only
  2215. Issue.update_all("assigned_to_id = NULL")
  2216. q = IssueQuery.new(:name => '_', :group_by => 'assigned_to')
  2217. count_by_group = q.result_count_by_group
  2218. assert_kind_of Hash, count_by_group
  2219. assert_equal 1, count_by_group.keys.size
  2220. assert_nil count_by_group.keys.first
  2221. end
  2222. def test_issue_ids
  2223. q = IssueQuery.new(:name => '_')
  2224. q.sort_criteria = ['subject', 'id']
  2225. issues = q.issues
  2226. assert_equal issues.map(&:id), q.issue_ids
  2227. end
  2228. def test_label_for
  2229. set_language_if_valid 'en'
  2230. q = IssueQuery.new
  2231. assert_equal 'Assignee', q.label_for('assigned_to_id')
  2232. end
  2233. def test_label_for_fr
  2234. set_language_if_valid 'fr'
  2235. q = IssueQuery.new
  2236. assert_equal 'Assigné à', q.label_for('assigned_to_id')
  2237. end
  2238. def test_editable_by
  2239. admin = User.find(1)
  2240. manager = User.find(2)
  2241. developer = User.find(3)
  2242. # Public query on project 1
  2243. q = IssueQuery.find(1)
  2244. assert q.editable_by?(admin)
  2245. assert q.editable_by?(manager)
  2246. assert !q.editable_by?(developer)
  2247. # Private query on project 1
  2248. q = IssueQuery.find(2)
  2249. assert q.editable_by?(admin)
  2250. assert !q.editable_by?(manager)
  2251. assert q.editable_by?(developer)
  2252. # Private query for all projects
  2253. q = IssueQuery.find(3)
  2254. assert q.editable_by?(admin)
  2255. assert !q.editable_by?(manager)
  2256. assert q.editable_by?(developer)
  2257. end
  2258. def test_editable_by_for_global_query
  2259. admin = User.find(1)
  2260. manager = User.find(2)
  2261. developer = User.find(3)
  2262. q = IssueQuery.find(4)
  2263. assert q.editable_by?(admin)
  2264. assert !q.editable_by?(manager)
  2265. assert !q.editable_by?(developer)
  2266. end
  2267. def test_editable_by_for_global_query_with_project_set
  2268. admin = User.find(1)
  2269. manager = User.find(2)
  2270. developer = User.find(3)
  2271. q = IssueQuery.find(4)
  2272. q.project = Project.find(1)
  2273. assert q.editable_by?(admin)
  2274. assert !q.editable_by?(manager)
  2275. assert !q.editable_by?(developer)
  2276. end
  2277. def test_visible_scope
  2278. query_ids = IssueQuery.visible(User.anonymous).map(&:id)
  2279. assert query_ids.include?(1), 'public query on public project was not visible'
  2280. assert query_ids.include?(4), 'public query for all projects was not visible'
  2281. assert !query_ids.include?(2), 'private query on public project was visible'
  2282. assert !query_ids.include?(3), 'private query for all projects was visible'
  2283. assert !query_ids.include?(7), 'public query on private project was visible'
  2284. end
  2285. def test_query_with_public_visibility_should_be_visible_to_anyone
  2286. q = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_PUBLIC)
  2287. assert q.visible?(User.anonymous)
  2288. assert IssueQuery.visible(User.anonymous).find_by_id(q.id)
  2289. assert q.visible?(User.find(7))
  2290. assert IssueQuery.visible(User.find(7)).find_by_id(q.id)
  2291. assert q.visible?(User.find(2))
  2292. assert IssueQuery.visible(User.find(2)).find_by_id(q.id)
  2293. assert q.visible?(User.find(1))
  2294. assert IssueQuery.visible(User.find(1)).find_by_id(q.id)
  2295. end
  2296. def test_query_with_roles_visibility_should_be_visible_to_user_with_role
  2297. q = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_ROLES, :role_ids => [1, 2])
  2298. assert !q.visible?(User.anonymous)
  2299. assert_nil IssueQuery.visible(User.anonymous).find_by_id(q.id)
  2300. assert !q.visible?(User.find(7))
  2301. assert_nil IssueQuery.visible(User.find(7)).find_by_id(q.id)
  2302. assert q.visible?(User.find(2))
  2303. assert IssueQuery.visible(User.find(2)).find_by_id(q.id)
  2304. assert q.visible?(User.find(1))
  2305. assert IssueQuery.visible(User.find(1)).find_by_id(q.id)
  2306. # Should ignore archived project memberships
  2307. Project.find(1).archive
  2308. assert !q.visible?(User.find(3))
  2309. assert_nil IssueQuery.visible(User.find(3)).find_by_id(q.id)
  2310. end
  2311. def test_query_with_private_visibility_should_be_visible_to_owner
  2312. q = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_PRIVATE, :user => User.find(7))
  2313. assert !q.visible?(User.anonymous)
  2314. assert_nil IssueQuery.visible(User.anonymous).find_by_id(q.id)
  2315. assert q.visible?(User.find(7))
  2316. assert IssueQuery.visible(User.find(7)).find_by_id(q.id)
  2317. assert !q.visible?(User.find(2))
  2318. assert_nil IssueQuery.visible(User.find(2)).find_by_id(q.id)
  2319. assert q.visible?(User.find(1))
  2320. assert_nil IssueQuery.visible(User.find(1)).find_by_id(q.id)
  2321. end
  2322. def test_build_from_params_should_not_update_query_with_nil_param_values
  2323. q =
  2324. IssueQuery.create!(
  2325. :name => 'Query',
  2326. :type => "IssueQuery",
  2327. :user => User.find(7),
  2328. :filters => {"status_id" => {:values => ["1"], :operator => "o"}},
  2329. :column_names => [:tracker, :status],
  2330. :sort_criteria => ['id', 'asc'],
  2331. :group_by => "project",
  2332. :options => {
  2333. :totalable_names=>[:estimated_hours],
  2334. :draw_relations => '1',
  2335. :draw_progress_line => '1'
  2336. }
  2337. )
  2338. old_attributes = q.attributes
  2339. q.build_from_params({})
  2340. assert_equal old_attributes, q.attributes
  2341. end
  2342. test "#available_filters should include users of visible projects in cross-project view" do
  2343. users = IssueQuery.new.available_filters["assigned_to_id"]
  2344. assert_not_nil users
  2345. assert users[:values].pluck(1).include?("3")
  2346. end
  2347. test "#available_filters should include users of subprojects" do
  2348. user1 = User.generate!
  2349. user2 = User.generate!
  2350. project = Project.find(1)
  2351. Member.create!(:principal => user1, :project => project.children.visible.first, :role_ids => [1])
  2352. users = IssueQuery.new(:project => project).available_filters["assigned_to_id"]
  2353. assert_not_nil users
  2354. assert users[:values].pluck(1).include?(user1.id.to_s)
  2355. assert !users[:values].pluck(1).include?(user2.id.to_s)
  2356. end
  2357. test "#available_filters should include visible projects in cross-project view" do
  2358. projects = IssueQuery.new.available_filters["project_id"]
  2359. assert_not_nil projects
  2360. assert projects[:values].pluck(1).include?("1")
  2361. end
  2362. test "#available_filters should include 'member_of_group' filter" do
  2363. query = IssueQuery.new
  2364. assert query.available_filters.key?("member_of_group")
  2365. assert_equal :list_optional, query.available_filters["member_of_group"][:type]
  2366. assert query.available_filters["member_of_group"][:values].present?
  2367. assert_equal Group.givable.sort.map {|g| [g.name, g.id.to_s]},
  2368. query.available_filters["member_of_group"][:values].sort
  2369. end
  2370. test "#available_filters should include 'assigned_to_role' filter" do
  2371. query = IssueQuery.new
  2372. assert query.available_filters.key?("assigned_to_role")
  2373. assert_equal :list_optional, query.available_filters["assigned_to_role"][:type]
  2374. assert query.available_filters["assigned_to_role"][:values].include?(['Manager', '1'])
  2375. assert query.available_filters["assigned_to_role"][:values].include?(['Developer', '2'])
  2376. assert query.available_filters["assigned_to_role"][:values].include?(['Reporter', '3'])
  2377. assert ! query.available_filters["assigned_to_role"][:values].include?(['Non member', '4'])
  2378. assert ! query.available_filters["assigned_to_role"][:values].include?(['Anonymous', '5'])
  2379. end
  2380. def test_available_filters_should_include_custom_field_according_to_user_visibility
  2381. visible_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => true)
  2382. hidden_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => false, :role_ids => [1])
  2383. with_current_user User.find(3) do
  2384. query = IssueQuery.new
  2385. assert_include "cf_#{visible_field.id}", query.available_filters.keys
  2386. assert_not_include "cf_#{hidden_field.id}", query.available_filters.keys
  2387. end
  2388. end
  2389. def test_available_columns_should_include_custom_field_according_to_user_visibility
  2390. visible_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => true)
  2391. hidden_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => false, :role_ids => [1])
  2392. with_current_user User.find(3) do
  2393. query = IssueQuery.new
  2394. assert_include :"cf_#{visible_field.id}", query.available_columns.map(&:name)
  2395. assert_not_include :"cf_#{hidden_field.id}", query.available_columns.map(&:name)
  2396. end
  2397. end
  2398. def test_available_columns_should_not_include_total_estimated_hours_when_trackers_disabled_estimated_hours
  2399. Tracker.visible.each do |tracker|
  2400. tracker.core_fields = tracker.core_fields.reject{|field| field == 'estimated_hours'}
  2401. tracker.save!
  2402. end
  2403. query = IssueQuery.new
  2404. available_columns = query.available_columns.map(&:name)
  2405. assert_not_include :estimated_hours, available_columns
  2406. assert_not_include :total_estimated_hours, available_columns
  2407. tracker = Tracker.visible.first
  2408. tracker.core_fields = ['estimated_hours']
  2409. tracker.save!
  2410. query = IssueQuery.new
  2411. available_columns = query.available_columns.map(&:name)
  2412. assert_include :estimated_hours, available_columns
  2413. assert_include :total_estimated_hours, available_columns
  2414. end
  2415. def setup_member_of_group
  2416. Group.destroy_all # No fixtures
  2417. @user_in_group = User.generate!
  2418. @second_user_in_group = User.generate!
  2419. @user_in_group2 = User.generate!
  2420. @user_not_in_group = User.generate!
  2421. @group = Group.generate!.reload
  2422. @group.users << @user_in_group
  2423. @group.users << @second_user_in_group
  2424. @group2 = Group.generate!.reload
  2425. @group2.users << @user_in_group2
  2426. @query = IssueQuery.new(:name => '_')
  2427. end
  2428. test "member_of_group filter should search assigned to for users in the group" do
  2429. setup_member_of_group
  2430. @query.add_filter('member_of_group', '=', [@group.id.to_s])
  2431. assert_find_issues_with_query_is_successful @query
  2432. end
  2433. test "member_of_group filter should search not assigned to any group member (none)" do
  2434. setup_member_of_group
  2435. @query.add_filter('member_of_group', '!*', [''])
  2436. assert_find_issues_with_query_is_successful @query
  2437. end
  2438. test "member_of_group filter should search assigned to any group member (all)" do
  2439. setup_member_of_group
  2440. @query.add_filter('member_of_group', '*', [''])
  2441. assert_find_issues_with_query_is_successful @query
  2442. end
  2443. test "member_of_group filter should return an empty set with = empty group" do
  2444. setup_member_of_group
  2445. @empty_group = Group.generate!
  2446. @query.add_filter('member_of_group', '=', [@empty_group.id.to_s])
  2447. assert_equal [], find_issues_with_query(@query)
  2448. end
  2449. test "member_of_group filter should return issues with ! empty group" do
  2450. setup_member_of_group
  2451. @empty_group = Group.generate!
  2452. @query.add_filter('member_of_group', '!', [@empty_group.id.to_s])
  2453. assert_find_issues_with_query_is_successful @query
  2454. end
  2455. def setup_assigned_to_role
  2456. @manager_role = Role.find_by_name('Manager')
  2457. @developer_role = Role.find_by_name('Developer')
  2458. @project = Project.generate!
  2459. @manager = User.generate!
  2460. @developer = User.generate!
  2461. @boss = User.generate!
  2462. @guest = User.generate!
  2463. User.add_to_project(@manager, @project, @manager_role)
  2464. User.add_to_project(@developer, @project, @developer_role)
  2465. User.add_to_project(@boss, @project, [@manager_role, @developer_role])
  2466. @issue1 = Issue.generate!(:project => @project, :assigned_to_id => @manager.id)
  2467. @issue2 = Issue.generate!(:project => @project, :assigned_to_id => @developer.id)
  2468. @issue3 = Issue.generate!(:project => @project, :assigned_to_id => @boss.id)
  2469. @issue4 = Issue.generate!(:project => @project, :author_id => @guest.id, :assigned_to_id => @guest.id)
  2470. @issue5 = Issue.generate!(:project => @project)
  2471. @query = IssueQuery.new(:name => '_', :project => @project)
  2472. end
  2473. test "assigned_to_role filter should search assigned to for users with the Role" do
  2474. setup_assigned_to_role
  2475. @query.add_filter('assigned_to_role', '=', [@manager_role.id.to_s])
  2476. assert_query_result [@issue1, @issue3], @query
  2477. end
  2478. test "assigned_to_role filter should search assigned to for users with the Role on the issue project" do
  2479. setup_assigned_to_role
  2480. other_project = Project.generate!
  2481. User.add_to_project(@developer, other_project, @manager_role)
  2482. @query.add_filter('assigned_to_role', '=', [@manager_role.id.to_s])
  2483. assert_query_result [@issue1, @issue3], @query
  2484. end
  2485. test "assigned_to_role filter should return an empty set with empty role" do
  2486. setup_assigned_to_role
  2487. @empty_role = Role.generate!
  2488. @query.add_filter('assigned_to_role', '=', [@empty_role.id.to_s])
  2489. assert_query_result [], @query
  2490. end
  2491. test "assigned_to_role filter should search assigned to for users without the Role" do
  2492. setup_assigned_to_role
  2493. @query.add_filter('assigned_to_role', '!', [@manager_role.id.to_s])
  2494. assert_query_result [@issue2, @issue4, @issue5], @query
  2495. end
  2496. test "assigned_to_role filter should search assigned to for users not assigned to any Role (none)" do
  2497. setup_assigned_to_role
  2498. @query.add_filter('assigned_to_role', '!*', [''])
  2499. assert_query_result [@issue4, @issue5], @query
  2500. end
  2501. test "assigned_to_role filter should search assigned to for users assigned to any Role (all)" do
  2502. setup_assigned_to_role
  2503. @query.add_filter('assigned_to_role', '*', [''])
  2504. assert_query_result [@issue1, @issue2, @issue3], @query
  2505. end
  2506. test "assigned_to_role filter should return issues with ! empty role" do
  2507. setup_assigned_to_role
  2508. @empty_role = Role.generate!
  2509. @query.add_filter('assigned_to_role', '!', [@empty_role.id.to_s])
  2510. assert_query_result [@issue1, @issue2, @issue3, @issue4, @issue5], @query
  2511. end
  2512. def test_query_column_should_accept_a_symbol_as_caption
  2513. set_language_if_valid 'en'
  2514. c = QueryColumn.new('foo', :caption => :general_text_Yes)
  2515. assert_equal 'Yes', c.caption
  2516. end
  2517. def test_query_column_should_accept_a_proc_as_caption
  2518. c = QueryColumn.new('foo', :caption => lambda {'Foo'})
  2519. assert_equal 'Foo', c.caption
  2520. end
  2521. def test_date_clause_should_respect_user_time_zone_with_local_default
  2522. @query = IssueQuery.new(:name => '_')
  2523. # user is in Hawaii (-10)
  2524. User.current = users(:users_001)
  2525. User.current.pref.update_attribute :time_zone, 'Hawaii'
  2526. # assume timestamps are stored in server local time
  2527. local_zone = Time.zone
  2528. from = Date.parse '2016-03-20'
  2529. to = Date.parse '2016-03-22'
  2530. assert c = @query.send(:date_clause, 'table', 'field', from, to, false)
  2531. # the dates should have been interpreted in the user's time zone and
  2532. # converted to local time
  2533. # what we get exactly in the sql depends on the local time zone, therefore
  2534. # it's computed here.
  2535. f = User.current.time_zone.local(from.year, from.month, from.day).yesterday.end_of_day.in_time_zone(local_zone)
  2536. t = User.current.time_zone.local(to.year, to.month, to.day).end_of_day.in_time_zone(local_zone)
  2537. assert_equal "table.field > '#{Query.connection.quoted_date f}' AND table.field <= '#{Query.connection.quoted_date t}'", c
  2538. end
  2539. def test_date_clause_should_respect_user_time_zone_with_utc_default
  2540. @query = IssueQuery.new(:name => '_')
  2541. # user is in Hawaii (-10)
  2542. User.current = users(:users_001)
  2543. User.current.pref.update_attribute :time_zone, 'Hawaii'
  2544. # assume timestamps are stored as utc
  2545. ActiveRecord.default_timezone = :utc
  2546. from = Date.parse '2016-03-20'
  2547. to = Date.parse '2016-03-22'
  2548. assert c = @query.send(:date_clause, 'table', 'field', from, to, false)
  2549. # the dates should have been interpreted in the user's time zone and
  2550. # converted to utc. March 20 in Hawaii begins at 10am UTC.
  2551. f = Time.new(2016, 3, 20, 9, 59, 59, 0).end_of_hour
  2552. t = Time.new(2016, 3, 23, 9, 59, 59, 0).end_of_hour
  2553. assert_equal "table.field > '#{Query.connection.quoted_date f}' AND table.field <= '#{Query.connection.quoted_date t}'", c
  2554. ensure
  2555. ActiveRecord.default_timezone = :local # restore Redmine default
  2556. end
  2557. def test_project_statement_with_closed_subprojects
  2558. project = Project.find(1)
  2559. project.descendants.each(&:close)
  2560. with_settings :display_subprojects_issues => '1' do
  2561. query = IssueQuery.new(:name => '_', :project => project)
  2562. statement = query.project_statement
  2563. assert_equal "projects.lft >= #{project.lft} AND projects.rgt <= #{project.rgt}", statement
  2564. end
  2565. end
  2566. def test_filter_on_subprojects
  2567. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  2568. filter_name = "subproject_id"
  2569. assert_include filter_name, query.available_filters.keys
  2570. # "is" operator should include issues of parent project + issues of the selected subproject
  2571. query.filters = {filter_name => {:operator => '=', :values => ['3']}}
  2572. issues = find_issues_with_query(query)
  2573. assert_equal [1, 2, 3, 5, 7, 8, 11, 12, 13, 14], issues.map(&:id).sort
  2574. # "is not" operator should include issues of parent project + issues of all active subprojects - issues of the selected subprojects
  2575. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  2576. query.filters = {filter_name => {:operator => '!', :values => ['3']}}
  2577. issues = find_issues_with_query(query)
  2578. assert_equal [1, 2, 3, 6, 7, 8, 9, 10, 11, 12], issues.map(&:id).sort
  2579. end
  2580. def test_filter_updated_on_none_should_return_issues_with_updated_on_equal_with_created_on
  2581. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  2582. query.filters = {'updated_on' => {:operator => '!*', :values => ['']}}
  2583. issues = find_issues_with_query(query)
  2584. assert_equal [3, 6, 7, 8, 9, 10, 14], issues.map(&:id).sort
  2585. end
  2586. def test_filter_updated_on_any_should_return_issues_with_updated_on_greater_than_created_on
  2587. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  2588. query.filters = {'updated_on' => {:operator => '*', :values => ['']}}
  2589. issues = find_issues_with_query(query)
  2590. assert_equal [1, 2, 5, 11, 12, 13], issues.map(&:id).sort
  2591. end
  2592. def test_issue_statuses_should_return_only_statuses_used_by_that_project
  2593. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  2594. query.filters = {'status_id' => {:operator => '=', :values => []}}
  2595. WorkflowTransition.delete_all
  2596. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 1, :new_status_id => 3)
  2597. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 1, :new_status_id => 4)
  2598. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 2, :new_status_id => 3)
  2599. WorkflowTransition.create(:role_id => 1, :tracker_id => 2, :old_status_id => 1, :new_status_id => 3)
  2600. assert_equal ['1', '2', '3', '4'], query.available_filters['status_id'][:values].map(&:second)
  2601. end
  2602. def test_issue_statuses_without_project_should_return_all_statuses
  2603. query = IssueQuery.new(:name => '_')
  2604. query.filters = {'status_id' => {:operator => '=', :values => []}}
  2605. WorkflowTransition.delete_all
  2606. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 1, :new_status_id => 3)
  2607. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 1, :new_status_id => 4)
  2608. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 2, :new_status_id => 3)
  2609. WorkflowTransition.create(:role_id => 1, :tracker_id => 2, :old_status_id => 1, :new_status_id => 3)
  2610. assert_equal ['1', '2', '3', '4', '5', '6'], query.available_filters['status_id'][:values].map(&:second)
  2611. end
  2612. def test_project_status_filter_should_be_available_in_global_queries
  2613. query = IssueQuery.new(:project => nil, :name => '_')
  2614. assert query.available_filters.has_key?('project.status')
  2615. end
  2616. def test_project_status_filter_should_be_available_when_project_has_subprojects
  2617. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  2618. assert query.available_filters.has_key?('project.status')
  2619. end
  2620. def test_project_status_filter_should_not_be_available_when_project_is_leaf
  2621. query = IssueQuery.new(:project => Project.find(2), :name => '_')
  2622. assert !query.available_filters.has_key?('project.status')
  2623. end
  2624. def test_project_statuses_values_should_return_only_active_and_closed_statuses
  2625. set_language_if_valid 'en'
  2626. query = IssueQuery.new(:project => nil, :name => '_')
  2627. project_status_filter = query.available_filters['project.status']
  2628. assert_not_nil project_status_filter
  2629. assert_equal [["active", "1"], ["closed", "5"]], project_status_filter[:values]
  2630. end
  2631. def test_as_params_should_serialize_query
  2632. query = IssueQuery.new(name: "_")
  2633. query.add_filter('subject', '!~', ['asdf'])
  2634. query.group_by = 'tracker'
  2635. query.totalable_names = %w(estimated_hours)
  2636. query.column_names = %w(id subject estimated_hours)
  2637. assert hsh = query.as_params
  2638. new_query = IssueQuery.build_from_params(hsh)
  2639. assert_equal query.filters, new_query.filters
  2640. assert_equal query.group_by, new_query.group_by
  2641. assert_equal query.column_names, new_query.column_names
  2642. assert_equal query.totalable_names, new_query.totalable_names
  2643. end
  2644. def test_issue_query_filter_by_spent_time
  2645. query = IssueQuery.new(:name => '_')
  2646. query.filters = {'spent_time' => {:operator => '*', :values => ['']}}
  2647. assert_equal [3, 1], query.issues.pluck(:id)
  2648. query.filters = {'spent_time' => {:operator => '!*', :values => ['']}}
  2649. assert_equal [13, 12, 11, 8, 7, 5, 2], query.issues.pluck(:id)
  2650. query.filters = {'spent_time' => {:operator => '>=', :values => ['10']}}
  2651. assert_equal [1], query.issues.pluck(:id)
  2652. query.filters = {'spent_time' => {:operator => '<=', :values => ['10']}}
  2653. assert_equal [13, 12, 11, 8, 7, 5, 3, 2], query.issues.pluck(:id)
  2654. query.filters = {'spent_time' => {:operator => '><', :values => ['1', '2']}}
  2655. assert_equal [3], query.issues.pluck(:id)
  2656. end
  2657. def test_issues_should_be_in_the_same_order_when_paginating
  2658. q = IssueQuery.new
  2659. q.sort_criteria = {'0' => ['priority', 'desc']}
  2660. issue_ids = q.issues.pluck(:id)
  2661. paginated_issue_ids = []
  2662. # Test with a maximum of 2 records per page.
  2663. ((q.issue_count / 2) + 1).times do |i|
  2664. paginated_issue_ids += q.issues(:offset => (i * 2), :limit => 2).pluck(:id)
  2665. end
  2666. # Non-paginated issue ids and paginated issue ids should be in the same order.
  2667. assert_equal issue_ids, paginated_issue_ids
  2668. end
  2669. def test_destruction_of_default_query_should_remove_reference_from_project
  2670. project = Project.find('ecookbook')
  2671. project_query = IssueQuery.find(1)
  2672. project.update_column :default_issue_query_id, project_query.id
  2673. project_query.destroy
  2674. project.reload
  2675. assert_nil project.default_issue_query_id
  2676. end
  2677. def test_should_determine_default_issue_query
  2678. project = Project.find('ecookbook')
  2679. user = project.users.first
  2680. project_query = IssueQuery.find(1)
  2681. query = IssueQuery.find(4)
  2682. user_query = IssueQuery.find(3)
  2683. user_query.update(visibility: Query::VISIBILITY_PUBLIC)
  2684. user_query.update_column :user_id, user.id
  2685. [nil, user, User.anonymous].each do |u|
  2686. [nil, project].each do |p|
  2687. assert_nil IssueQuery.default(project: p, user: u)
  2688. end
  2689. end
  2690. # only global default is set
  2691. with_settings :default_issue_query => query.id do
  2692. [nil, user, User.anonymous].each do |u|
  2693. [nil, project].each do |p|
  2694. assert_equal query, IssueQuery.default(project: p, user: u)
  2695. end
  2696. end
  2697. end
  2698. # with project default
  2699. assert_equal project.id, project_query.project_id
  2700. project.update_column :default_issue_query_id, project_query.id
  2701. [nil, user, User.anonymous].each do |u|
  2702. assert_nil IssueQuery.default(project: nil, user: u)
  2703. assert_equal project_query, IssueQuery.default(project: project, user: u)
  2704. end
  2705. # project default should override global default
  2706. with_settings :default_issue_query => query.id do
  2707. [nil, user, User.anonymous].each do |u|
  2708. assert_equal query, IssueQuery.default(project: nil, user: u)
  2709. assert_equal project_query, IssueQuery.default(project: project, user: u)
  2710. end
  2711. end
  2712. # user default, overrides project and global default
  2713. user.pref.default_issue_query = user_query.id
  2714. user.pref.save
  2715. with_settings :default_issue_query => query.id do
  2716. [nil, project].each do |p|
  2717. assert_equal user_query, IssueQuery.default(project: p, user: user)
  2718. assert_equal user_query, IssueQuery.default(project: p, user: user)
  2719. end
  2720. end
  2721. end
  2722. def test_sql_contains_should_escape_value
  2723. i = Issue.generate! subject: 'Sanitize test'
  2724. query = IssueQuery.new(:project => nil, :name => '_')
  2725. query.add_filter('subject', '~', ['te%t'])
  2726. assert_equal 0, query.issue_count
  2727. i.update_column :subject, 'Sanitize te%t'
  2728. assert_equal 1, query.issue_count
  2729. i.update_column :subject, 'Sanitize te_t'
  2730. query = IssueQuery.new(:project => nil, :name => '_')
  2731. query.add_filter('subject', '~', ['te_t'])
  2732. assert_equal 1, query.issue_count
  2733. end
  2734. def test_sql_contains_should_tokenize
  2735. query = IssueQuery.new(:project => nil, :name => '_')
  2736. query.add_filter('subject', '~', ['issue today'])
  2737. assert_equal 1, query.issue_count
  2738. end
  2739. def test_sql_contains_should_tokenize_for_starts_with
  2740. query = IssueQuery.new(
  2741. :project => nil, :name => '_',
  2742. :filters => {
  2743. 'subject' => {:operator => '^', :values => ['issue closed']}
  2744. }
  2745. )
  2746. assert_equal 4, query.issue_count
  2747. query.issues.each do |issue|
  2748. assert_match /^(issue|closed)/i, issue.subject
  2749. end
  2750. end
  2751. def test_sql_contains_should_tokenize_for_ends_with
  2752. query = IssueQuery.new(
  2753. :project => nil, :name => '_',
  2754. :filters => {
  2755. 'subject' => {:operator => '$', :values => ['version issue']}
  2756. }
  2757. )
  2758. assert_equal 4, query.issue_count
  2759. query.issues.each do |issue|
  2760. assert_match /(version|issue)$/i, issue.subject
  2761. end
  2762. end
  2763. def test_display_type_should_accept_known_types
  2764. query = ProjectQuery.new(:name => '_')
  2765. query.display_type = 'list'
  2766. assert_equal 'list', query.display_type
  2767. end
  2768. def test_display_type_should_not_accept_unknown_types
  2769. query = ProjectQuery.new(:name => '_')
  2770. query.display_type = 'invalid'
  2771. assert_equal 'board', query.display_type
  2772. end
  2773. end