您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

query_test.rb 111KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897
  1. # frozen_string_literal: true
  2. # Redmine - project management software
  3. # Copyright (C) 2006-2023 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_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_all
  198. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  199. query.add_filter('fixed_version_id', '*', [''])
  200. query.add_filter('cf_1', '*', [''])
  201. assert query.statement.include?("#{Issue.table_name}.fixed_version_id IS NOT NULL")
  202. assert query.statement.include?("#{CustomValue.table_name}.value IS NOT NULL AND #{CustomValue.table_name}.value <> ''")
  203. find_issues_with_query(query)
  204. end
  205. def test_operator_all_for_date
  206. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  207. query.add_filter('start_date', '*', [''])
  208. issues = find_issues_with_query(query)
  209. assert !issues.empty?
  210. assert issues.all? {|i| i.start_date.present?}
  211. end
  212. def test_operator_all_for_string_custom_field
  213. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  214. query.add_filter('cf_2', '*', [''])
  215. assert query.has_filter?('cf_2')
  216. issues = find_issues_with_query(query)
  217. assert !issues.empty?
  218. assert issues.all? {|i| i.custom_field_value(2).present?}
  219. end
  220. def test_numeric_filter_should_not_accept_non_numeric_values
  221. query = IssueQuery.new(:name => '_')
  222. query.add_filter('estimated_hours', '=', ['a'])
  223. assert query.has_filter?('estimated_hours')
  224. assert !query.valid?
  225. end
  226. def test_operator_is_on_float
  227. Issue.where(:id => 2).update_all("estimated_hours = 171.2")
  228. query = IssueQuery.new(:name => '_')
  229. query.add_filter('estimated_hours', '=', ['171.20'])
  230. issues = find_issues_with_query(query)
  231. assert_equal 1, issues.size
  232. assert_equal 2, issues.first.id
  233. end
  234. def test_operator_is_on_issue_id_should_accept_comma_separated_values
  235. query = IssueQuery.new(:name => '_')
  236. query.add_filter("issue_id", '=', ['1,3'])
  237. issues = find_issues_with_query(query)
  238. assert_equal 2, issues.size
  239. assert_equal [1, 3], issues.map(&:id).sort
  240. end
  241. def test_operator_is_on_parent_id_should_accept_comma_separated_values
  242. Issue.where(:id => [2, 4]).update_all(:parent_id => 1)
  243. Issue.where(:id => 5).update_all(:parent_id => 3)
  244. query = IssueQuery.new(:name => '_')
  245. query.add_filter("parent_id", '=', ['1,3'])
  246. issues = find_issues_with_query(query)
  247. assert_equal 3, issues.size
  248. assert_equal [2, 4, 5], issues.map(&:id).sort
  249. end
  250. def test_operator_is_on_child_id_should_accept_comma_separated_values
  251. Issue.where(:id => [2, 4]).update_all(:parent_id => 1)
  252. Issue.where(:id => 5).update_all(:parent_id => 3)
  253. query = IssueQuery.new(:name => '_')
  254. query.add_filter("child_id", '=', ['2,4,5'])
  255. issues = find_issues_with_query(query)
  256. assert_equal 2, issues.size
  257. assert_equal [1, 3], issues.map(&:id).sort
  258. end
  259. def test_operator_between_on_issue_id_should_return_range
  260. query = IssueQuery.new(:name => '_')
  261. query.add_filter("issue_id", '><', ['2', '3'])
  262. issues = find_issues_with_query(query)
  263. assert_equal 2, issues.size
  264. assert_equal [2, 3], issues.map(&:id).sort
  265. end
  266. def test_operator_is_on_integer_custom_field
  267. f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_for_all => true, :is_filter => true, :trackers => Tracker.all)
  268. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7')
  269. CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12')
  270. CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
  271. query = IssueQuery.new(:name => '_')
  272. query.add_filter("cf_#{f.id}", '=', ['12'])
  273. issues = find_issues_with_query(query)
  274. assert_equal 1, issues.size
  275. assert_equal 2, issues.first.id
  276. end
  277. def test_operator_is_on_integer_custom_field_should_accept_negative_value
  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. assert query.valid?
  285. issues = find_issues_with_query(query)
  286. assert_equal 1, issues.size
  287. assert_equal 2, issues.first.id
  288. end
  289. def test_operator_is_on_float_custom_field
  290. f = IssueCustomField.create!(:name => 'filter', :field_format => 'float', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
  291. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7.3')
  292. CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12.7')
  293. CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
  294. query = IssueQuery.new(:name => '_')
  295. query.add_filter("cf_#{f.id}", '=', ['12.7'])
  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_should_accept_negative_value
  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. assert query.valid?
  308. issues = find_issues_with_query(query)
  309. assert_equal 1, issues.size
  310. assert_equal 2, issues.first.id
  311. end
  312. def test_operator_is_on_multi_list_custom_field
  313. f = IssueCustomField.create!(:name => 'filter', :field_format => 'list', :is_filter => true, :is_for_all => true,
  314. :possible_values => ['value1', 'value2', 'value3'], :multiple => true, :trackers => Tracker.all)
  315. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value1')
  316. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value2')
  317. CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => 'value1')
  318. query = IssueQuery.new(:name => '_')
  319. query.add_filter("cf_#{f.id}", '=', ['value1'])
  320. issues = find_issues_with_query(query)
  321. assert_equal [1, 3], issues.map(&:id).sort
  322. query = IssueQuery.new(:name => '_')
  323. query.add_filter("cf_#{f.id}", '=', ['value2'])
  324. issues = find_issues_with_query(query)
  325. assert_equal [1], issues.map(&:id).sort
  326. end
  327. def test_operator_is_not_on_multi_list_custom_field
  328. f = IssueCustomField.create!(:name => 'filter', :field_format => 'list', :is_filter => true, :is_for_all => true,
  329. :possible_values => ['value1', 'value2', 'value3'], :multiple => true, :trackers => Tracker.all)
  330. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value1')
  331. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'value2')
  332. CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => 'value1')
  333. query = IssueQuery.new(:name => '_')
  334. query.add_filter("cf_#{f.id}", '!', ['value1'])
  335. issues = find_issues_with_query(query)
  336. assert !issues.map(&:id).include?(1)
  337. assert !issues.map(&:id).include?(3)
  338. query = IssueQuery.new(:name => '_')
  339. query.add_filter("cf_#{f.id}", '!', ['value2'])
  340. issues = find_issues_with_query(query)
  341. assert !issues.map(&:id).include?(1)
  342. assert issues.map(&:id).include?(3)
  343. end
  344. def test_operator_is_on_string_custom_field_with_utf8_value
  345. f = IssueCustomField.create!(:name => 'filter', :field_format => 'string', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
  346. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => 'Kiểm')
  347. query = IssueQuery.new(:name => '_')
  348. query.add_filter("cf_#{f.id}", '=', ['Kiểm'])
  349. issues = find_issues_with_query(query)
  350. assert_equal [1], issues.map(&:id).sort
  351. end
  352. def test_operator_is_on_is_private_field
  353. # is_private filter only available for those who can set issues private
  354. User.current = User.find(2)
  355. query = IssueQuery.new(:name => '_')
  356. assert query.available_filters.key?('is_private')
  357. query.add_filter("is_private", '=', ['1'])
  358. issues = find_issues_with_query(query)
  359. assert issues.any?
  360. assert_nil issues.detect {|issue| !issue.is_private?}
  361. end
  362. def test_operator_is_not_on_is_private_field
  363. # is_private filter only available for those who can set issues private
  364. User.current = User.find(2)
  365. query = IssueQuery.new(:name => '_')
  366. assert query.available_filters.key?('is_private')
  367. query.add_filter("is_private", '!', ['1'])
  368. issues = find_issues_with_query(query)
  369. assert issues.any?
  370. assert_nil issues.detect {|issue| issue.is_private?}
  371. end
  372. def test_operator_greater_than
  373. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  374. query.add_filter('done_ratio', '>=', ['40'])
  375. assert query.statement.include?("#{Issue.table_name}.done_ratio >= 40.0")
  376. find_issues_with_query(query)
  377. end
  378. def test_operator_greater_than_a_float
  379. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  380. query.add_filter('estimated_hours', '>=', ['40.5'])
  381. assert query.statement.include?("#{Issue.table_name}.estimated_hours >= 40.5")
  382. find_issues_with_query(query)
  383. end
  384. def test_operator_greater_than_on_int_custom_field
  385. f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
  386. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '7')
  387. CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '12')
  388. CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
  389. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  390. query.add_filter("cf_#{f.id}", '>=', ['8'])
  391. issues = find_issues_with_query(query)
  392. assert_equal 1, issues.size
  393. assert_equal 2, issues.first.id
  394. end
  395. def test_operator_lesser_than
  396. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  397. query.add_filter('done_ratio', '<=', ['30'])
  398. assert query.statement.include?("#{Issue.table_name}.done_ratio <= 30.0")
  399. find_issues_with_query(query)
  400. end
  401. def test_operator_lesser_than_on_custom_field
  402. f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
  403. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  404. query.add_filter("cf_#{f.id}", '<=', ['30'])
  405. assert_match /CAST.+ <= 30\.0/, query.statement
  406. find_issues_with_query(query)
  407. end
  408. def test_operator_lesser_than_on_date_custom_field
  409. f = IssueCustomField.create!(:name => 'filter', :field_format => 'date', :is_filter => true, :is_for_all => true, :trackers => Tracker.all)
  410. CustomValue.create!(:custom_field => f, :customized => Issue.find(1), :value => '2013-04-11')
  411. CustomValue.create!(:custom_field => f, :customized => Issue.find(2), :value => '2013-05-14')
  412. CustomValue.create!(:custom_field => f, :customized => Issue.find(3), :value => '')
  413. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  414. query.add_filter("cf_#{f.id}", '<=', ['2013-05-01'])
  415. issue_ids = find_issues_with_query(query).map(&:id)
  416. assert_include 1, issue_ids
  417. assert_not_include 2, issue_ids
  418. assert_not_include 3, issue_ids
  419. end
  420. def test_operator_between
  421. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  422. query.add_filter('done_ratio', '><', ['30', '40'])
  423. assert_include "#{Issue.table_name}.done_ratio BETWEEN 30.0 AND 40.0", query.statement
  424. find_issues_with_query(query)
  425. end
  426. def test_operator_between_on_custom_field
  427. f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
  428. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  429. query.add_filter("cf_#{f.id}", '><', ['30', '40'])
  430. assert_match /CAST.+ BETWEEN 30.0 AND 40.0/, query.statement
  431. find_issues_with_query(query)
  432. end
  433. def test_date_filter_should_not_accept_non_date_values
  434. query = IssueQuery.new(:name => '_')
  435. query.add_filter('created_on', '=', ['a'])
  436. assert query.has_filter?('created_on')
  437. assert !query.valid?
  438. end
  439. def test_date_filter_should_not_accept_invalid_date_values
  440. query = IssueQuery.new(:name => '_')
  441. query.add_filter('created_on', '=', ['2011-01-34'])
  442. assert query.has_filter?('created_on')
  443. assert !query.valid?
  444. end
  445. def test_relative_date_filter_should_not_accept_non_integer_values
  446. query = IssueQuery.new(:name => '_')
  447. query.add_filter('created_on', '>t-', ['a'])
  448. assert query.has_filter?('created_on')
  449. assert !query.valid?
  450. end
  451. def test_operator_date_equals
  452. query = IssueQuery.new(:name => '_')
  453. query.add_filter('due_date', '=', ['2011-07-10'])
  454. 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+)?/,
  455. query.statement
  456. find_issues_with_query(query)
  457. end
  458. def test_operator_date_lesser_than
  459. query = IssueQuery.new(:name => '_')
  460. query.add_filter('due_date', '<=', ['2011-07-10'])
  461. assert_match /issues\.due_date <= '#{quoted_date "2011-07-10"} 23:59:59(\.\d+)?/, query.statement
  462. find_issues_with_query(query)
  463. end
  464. def test_operator_date_lesser_than_with_timestamp
  465. query = IssueQuery.new(:name => '_')
  466. query.add_filter('updated_on', '<=', ['2011-07-10T19:13:52'])
  467. assert_match /issues\.updated_on <= '#{quoted_date "2011-07-10"} 19:13:52/, query.statement
  468. find_issues_with_query(query)
  469. end
  470. def test_operator_date_greater_than
  471. query = IssueQuery.new(:name => '_')
  472. query.add_filter('due_date', '>=', ['2011-07-10'])
  473. assert_match /issues\.due_date > '#{quoted_date "2011-07-09"} 23:59:59(\.\d+)?'/, query.statement
  474. find_issues_with_query(query)
  475. end
  476. def test_operator_date_greater_than_with_timestamp
  477. query = IssueQuery.new(:name => '_')
  478. query.add_filter('updated_on', '>=', ['2011-07-10T19:13:52'])
  479. assert_match /issues\.updated_on > '#{quoted_date "2011-07-10"} 19:13:51(\.0+)?'/, query.statement
  480. find_issues_with_query(query)
  481. end
  482. def test_operator_date_between
  483. query = IssueQuery.new(:name => '_')
  484. query.add_filter('due_date', '><', ['2011-06-23', '2011-07-10'])
  485. 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+)?'/,
  486. query.statement
  487. find_issues_with_query(query)
  488. end
  489. def test_operator_in_more_than
  490. Issue.find(7).update_attribute(:due_date, (Date.today + 15))
  491. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  492. query.add_filter('due_date', '>t+', ['15'])
  493. issues = find_issues_with_query(query)
  494. assert !issues.empty?
  495. issues.each {|issue| assert(issue.due_date >= (Date.today + 15))}
  496. end
  497. def test_operator_in_less_than
  498. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  499. query.add_filter('due_date', '<t+', ['15'])
  500. issues = find_issues_with_query(query)
  501. assert !issues.empty?
  502. issues.each {|issue| assert(issue.due_date <= (Date.today + 15))}
  503. end
  504. def test_operator_in_the_next_days
  505. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  506. query.add_filter('due_date', '><t+', ['15'])
  507. issues = find_issues_with_query(query)
  508. assert !issues.empty?
  509. issues.each {|issue| assert(issue.due_date >= Date.today && issue.due_date <= (Date.today + 15))}
  510. end
  511. def test_operator_less_than_ago
  512. Issue.find(7).update_attribute(:due_date, (Date.today - 3))
  513. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  514. query.add_filter('due_date', '>t-', ['3'])
  515. issues = find_issues_with_query(query)
  516. assert !issues.empty?
  517. issues.each {|issue| assert(issue.due_date >= (Date.today - 3))}
  518. end
  519. def test_operator_in_the_past_days
  520. Issue.find(7).update_attribute(:due_date, (Date.today - 3))
  521. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  522. query.add_filter('due_date', '><t-', ['3'])
  523. issues = find_issues_with_query(query)
  524. assert !issues.empty?
  525. issues.each {|issue| assert(issue.due_date >= (Date.today - 3) && issue.due_date <= Date.today)}
  526. end
  527. def test_operator_more_than_ago
  528. Issue.find(7).update_attribute(:due_date, (Date.today - 10))
  529. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  530. query.add_filter('due_date', '<t-', ['10'])
  531. assert query.statement.include?("#{Issue.table_name}.due_date <=")
  532. issues = find_issues_with_query(query)
  533. assert !issues.empty?
  534. issues.each {|issue| assert(issue.due_date <= (Date.today - 10))}
  535. end
  536. def test_operator_in
  537. Issue.find(7).update_attribute(:due_date, (Date.today + 2))
  538. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  539. query.add_filter('due_date', 't+', ['2'])
  540. issues = find_issues_with_query(query)
  541. assert !issues.empty?
  542. issues.each {|issue| assert_equal((Date.today + 2), issue.due_date)}
  543. end
  544. def test_operator_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_equal((Date.today - 3), issue.due_date)}
  551. end
  552. def test_operator_today
  553. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  554. query.add_filter('due_date', 't', [''])
  555. issues = find_issues_with_query(query)
  556. assert !issues.empty?
  557. issues.each {|issue| assert_equal Date.today, issue.due_date}
  558. end
  559. def test_operator_tomorrow
  560. issue = Issue.generate!(:due_date => User.current.today.tomorrow)
  561. other_issues = []
  562. other_issues << Issue.generate!(:due_date => User.current.today.yesterday)
  563. other_issues << Issue.generate!(:due_date => User.current.today + 2)
  564. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  565. query.add_filter('due_date', 'nd', [''])
  566. issues = find_issues_with_query(query)
  567. assert_include issue, issues
  568. other_issues.each {|i| assert_not_include i, issues}
  569. end
  570. def test_operator_date_periods
  571. %w(t ld w lw l2w m lm y nd nw nm).each do |operator|
  572. query = IssueQuery.new(:name => '_')
  573. query.add_filter('due_date', operator, [''])
  574. assert query.valid?
  575. assert query.issues
  576. end
  577. end
  578. def test_operator_datetime_periods
  579. %w(t ld w lw l2w m lm y).each do |operator|
  580. query = IssueQuery.new(:name => '_')
  581. query.add_filter('created_on', operator, [''])
  582. assert query.valid?
  583. assert query.issues
  584. end
  585. end
  586. def test_operator_contains
  587. issue = Issue.generate!(:subject => 'AbCdEfG')
  588. query = IssueQuery.new(:name => '_')
  589. query.add_filter('subject', '~', ['cdeF'])
  590. result = find_issues_with_query(query)
  591. assert_include issue, result
  592. result.each {|issue| assert issue.subject.downcase.include?('cdef')}
  593. end
  594. def test_operator_contains_with_utf8_string
  595. issue = Issue.generate!(:subject => 'Subject contains Kiểm')
  596. query = IssueQuery.new(:name => '_')
  597. query.add_filter('subject', '~', ['Kiểm'])
  598. result = find_issues_with_query(query)
  599. assert_include issue, result
  600. assert_equal 1, result.size
  601. end
  602. def test_operator_does_not_contain
  603. issue = Issue.generate!(:subject => 'AbCdEfG')
  604. query = IssueQuery.new(:name => '_')
  605. query.add_filter('subject', '!~', ['cdeF'])
  606. result = find_issues_with_query(query)
  607. assert_not_include issue, result
  608. end
  609. def test_range_for_this_week_with_week_starting_on_monday
  610. I18n.locale = :fr
  611. assert_equal '1', I18n.t(:general_first_day_of_week)
  612. Date.stubs(:today).returns(Date.parse('2011-04-29'))
  613. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  614. query.add_filter('due_date', 'w', [''])
  615. 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+)?/,
  616. query.statement
  617. I18n.locale = :en
  618. end
  619. def test_range_for_this_week_with_week_starting_on_sunday
  620. I18n.locale = :en
  621. assert_equal '7', I18n.t(:general_first_day_of_week)
  622. Date.stubs(:today).returns(Date.parse('2011-04-29'))
  623. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  624. query.add_filter('due_date', 'w', [''])
  625. 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+)?/,
  626. query.statement
  627. end
  628. def test_range_for_next_week_with_week_starting_on_monday
  629. I18n.locale = :fr
  630. assert_equal '1', I18n.t(:general_first_day_of_week)
  631. Date.stubs(:today).returns(Date.parse('2011-04-29')) # Friday
  632. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  633. query.add_filter('due_date', 'nw', [''])
  634. 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+)?/,
  635. query.statement
  636. I18n.locale = :en
  637. end
  638. def test_range_for_next_week_with_week_starting_on_sunday
  639. I18n.locale = :en
  640. assert_equal '7', I18n.t(:general_first_day_of_week)
  641. Date.stubs(:today).returns(Date.parse('2011-04-29')) # Friday
  642. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  643. query.add_filter('due_date', 'nw', [''])
  644. 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+)?/,
  645. query.statement
  646. end
  647. def test_range_for_next_month
  648. Date.stubs(:today).returns(Date.parse('2011-04-29')) # Friday
  649. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  650. query.add_filter('due_date', 'nm', [''])
  651. 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+)?/,
  652. query.statement
  653. end
  654. def test_filter_assigned_to_me
  655. user = User.find(2)
  656. group = Group.find(10)
  657. group.users << user
  658. other_group = Group.find(11)
  659. Member.create!(:project_id => 1, :principal => group, :role_ids => [1])
  660. Member.create!(:project_id => 1, :principal => other_group, :role_ids => [1])
  661. User.current = user
  662. with_settings :issue_group_assignment => '1' do
  663. i1 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => user)
  664. i2 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => group)
  665. i3 = Issue.generate!(:project_id => 1, :tracker_id => 1, :assigned_to => other_group)
  666. query =
  667. IssueQuery.new(
  668. :name => '_',
  669. :filters => {
  670. 'assigned_to_id' => {
  671. :operator => '=',
  672. :values => ['me']
  673. }
  674. }
  675. )
  676. result = query.issues
  677. assert_equal(
  678. Issue.visible.where(:assigned_to_id => ([2] + user.reload.group_ids)).sort_by(&:id),
  679. result.sort_by(&:id)
  680. )
  681. assert result.include?(i1)
  682. assert result.include?(i2)
  683. assert !result.include?(i3)
  684. end
  685. end
  686. def test_filter_notes
  687. user = User.generate!
  688. Journal.create!(:user_id => user.id, :journalized => Issue.find(2), :notes => 'Notes.')
  689. Journal.create!(:user_id => user.id, :journalized => Issue.find(3), :notes => 'Notes.')
  690. issue_journals = Issue.find(1).journals.sort
  691. assert_equal ['Journal notes', 'Some notes with Redmine links: #2, r2.'], issue_journals.map(&:notes)
  692. assert_equal [false, false], issue_journals.map(&:private_notes)
  693. query = IssueQuery.new(:name => '_')
  694. filter_name = 'notes'
  695. assert_include filter_name, query.available_filters.keys
  696. {
  697. '~' => [1, 2, 3],
  698. '!~' => Issue.ids.sort - [1, 2, 3],
  699. '^' => [2, 3],
  700. '$' => [1],
  701. }.each do |operator, expected|
  702. query.filters = {filter_name => {:operator => operator, :values => ['Notes']}}
  703. assert_equal expected, find_issues_with_query(query).map(&:id).sort
  704. end
  705. end
  706. def test_filter_notes_should_ignore_private_notes_that_are_not_visible
  707. user = User.generate!
  708. Journal.create!(:user_id => user.id, :journalized => Issue.find(2), :notes => 'Notes.', :private_notes => true)
  709. Journal.create!(:user_id => user.id, :journalized => Issue.find(3), :notes => 'Notes.')
  710. issue_journals = Issue.find(1).journals.sort
  711. assert_equal ['Journal notes', 'Some notes with Redmine links: #2, r2.'], issue_journals.map(&:notes)
  712. assert_equal [false, false], issue_journals.map(&:private_notes)
  713. query = IssueQuery.new(:name => '_')
  714. filter_name = 'notes'
  715. assert_include filter_name, query.available_filters.keys
  716. query.filters = {filter_name => {:operator => '~', :values => ['Notes']}}
  717. assert_equal [1, 3], find_issues_with_query(query).map(&:id).sort
  718. end
  719. def test_filter_updated_by
  720. user = User.generate!
  721. Journal.create!(:user_id => user.id, :journalized => Issue.find(2), :notes => 'Notes')
  722. Journal.create!(:user_id => user.id, :journalized => Issue.find(3), :notes => 'Notes')
  723. Journal.create!(:user_id => 2, :journalized => Issue.find(3), :notes => 'Notes')
  724. query = IssueQuery.new(:name => '_')
  725. filter_name = "updated_by"
  726. assert_include filter_name, query.available_filters.keys
  727. query.filters = {filter_name => {:operator => '=', :values => [user.id]}}
  728. assert_equal [2, 3], find_issues_with_query(query).map(&:id).sort
  729. query.filters = {filter_name => {:operator => '!', :values => [user.id]}}
  730. assert_equal (Issue.ids.sort - [2, 3]), find_issues_with_query(query).map(&:id).sort
  731. end
  732. def test_filter_updated_by_should_ignore_private_notes_that_are_not_visible
  733. user = User.generate!
  734. Journal.create!(:user_id => user.id, :journalized => Issue.find(2), :notes => 'Notes', :private_notes => true)
  735. Journal.create!(:user_id => user.id, :journalized => Issue.find(3), :notes => 'Notes')
  736. query = IssueQuery.new(:name => '_')
  737. filter_name = "updated_by"
  738. assert_include filter_name, query.available_filters.keys
  739. with_current_user User.anonymous do
  740. query.filters = {filter_name => {:operator => '=', :values => [user.id]}}
  741. assert_equal [3], find_issues_with_query(query).map(&:id).sort
  742. end
  743. end
  744. def test_filter_updated_by_me
  745. user = User.generate!
  746. Journal.create!(:user_id => user.id, :journalized => Issue.find(2), :notes => 'Notes')
  747. with_current_user user do
  748. query = IssueQuery.new(:name => '_')
  749. filter_name = "updated_by"
  750. assert_include filter_name, query.available_filters.keys
  751. query.filters = {filter_name => {:operator => '=', :values => ['me']}}
  752. assert_equal [2], find_issues_with_query(query).map(&:id).sort
  753. end
  754. end
  755. def test_filter_last_updated_by
  756. user = User.generate!
  757. Journal.create!(:user_id => user.id, :journalized => Issue.find(2), :notes => 'Notes')
  758. Journal.create!(:user_id => user.id, :journalized => Issue.find(3), :notes => 'Notes')
  759. Journal.create!(:user_id => 2, :journalized => Issue.find(3), :notes => 'Notes')
  760. query = IssueQuery.new(:name => '_')
  761. filter_name = "last_updated_by"
  762. assert_include filter_name, query.available_filters.keys
  763. query.filters = {filter_name => {:operator => '=', :values => [user.id]}}
  764. assert_equal [2], find_issues_with_query(query).map(&:id).sort
  765. end
  766. def test_filter_last_updated_by_should_ignore_private_notes_that_are_not_visible
  767. user1 = User.generate!
  768. user2 = User.generate!
  769. Journal.create!(:user_id => user1.id, :journalized => Issue.find(2), :notes => 'Notes')
  770. Journal.create!(:user_id => user2.id, :journalized => Issue.find(2), :notes => 'Notes', :private_notes => true)
  771. query = IssueQuery.new(:name => '_')
  772. filter_name = "last_updated_by"
  773. assert_include filter_name, query.available_filters.keys
  774. with_current_user User.anonymous do
  775. query.filters = {filter_name => {:operator => '=', :values => [user1.id]}}
  776. assert_equal [2], find_issues_with_query(query).map(&:id).sort
  777. query.filters = {filter_name => {:operator => '=', :values => [user2.id]}}
  778. assert_equal [], find_issues_with_query(query).map(&:id).sort
  779. end
  780. with_current_user User.find(2) do
  781. query.filters = {filter_name => {:operator => '=', :values => [user1.id]}}
  782. assert_equal [], find_issues_with_query(query).map(&:id).sort
  783. query.filters = {filter_name => {:operator => '=', :values => [user2.id]}}
  784. assert_equal [2], find_issues_with_query(query).map(&:id).sort
  785. end
  786. end
  787. def test_user_custom_field_filtered_on_me
  788. User.current = User.find(2)
  789. cf = IssueCustomField.create!(:field_format => 'user', :is_for_all => true, :is_filter => true, :name => 'User custom field', :tracker_ids => [1])
  790. issue1 = Issue.create!(:project_id => 1, :tracker_id => 1, :custom_field_values => {cf.id.to_s => '2'}, :subject => 'Test', :author_id => 1)
  791. issue2 = Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {cf.id.to_s => '3'})
  792. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  793. filter = query.available_filters["cf_#{cf.id}"]
  794. assert_not_nil filter
  795. assert_include 'me', filter[:values].pluck(1)
  796. query.filters = {"cf_#{cf.id}" => {:operator => '=', :values => ['me']}}
  797. result = query.issues
  798. assert_equal 1, result.size
  799. assert_equal issue1, result.first
  800. end
  801. def test_filter_on_chained_user_custom_field
  802. user = User.find(2)
  803. User.current = user
  804. user_cf = UserCustomField.find(4)
  805. user_cf.update! is_filter: true
  806. issue_cf = IssueCustomField.create!(:field_format => 'user', :is_for_all => true, :is_filter => true, :name => 'User custom field', :tracker_ids => [1])
  807. issue1 = Issue.create!(:project_id => 1, :tracker_id => 1, :custom_field_values => {issue_cf.id.to_s => '2'}, :subject => 'Test', :author_id => 1)
  808. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  809. query.filters = {"cf_#{issue_cf.id}.cf_#{user_cf.id}" => {:operator => '~', :values => ['01 42']}}
  810. result = query.issues
  811. assert_equal 1, result.size
  812. assert_equal issue1, result.first
  813. end
  814. def test_filter_on_chained_user_custom_field_of_type_float
  815. user_cf = UserCustomField.find(5)
  816. user_cf.update! is_filter: true
  817. issue_cf = IssueCustomField.create!(:field_format => 'user', :is_for_all => true, :is_filter => true, :name => 'User custom field', :tracker_ids => [1])
  818. issue1 = Issue.create!(:project_id => 1, :tracker_id => 1, :custom_field_values => {issue_cf.id.to_s => '2'}, :subject => 'Test', :author_id => 1)
  819. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  820. query.filters = {"cf_#{issue_cf.id}.cf_#{user_cf.id}" => {:operator => '=', :values => ["30.1"]}}
  821. assert query.issues
  822. end
  823. def test_filter_on_me_by_anonymous_user
  824. User.current = nil
  825. query =
  826. IssueQuery.new(
  827. :name => '_',
  828. :filters => {
  829. 'assigned_to_id' => {
  830. :operator => '=',
  831. :values => ['me']
  832. }
  833. }
  834. )
  835. assert_equal [], query.issues
  836. end
  837. def test_filter_my_projects
  838. User.current = User.find(2)
  839. query = IssueQuery.new(:name => '_')
  840. filter = query.available_filters['project_id']
  841. assert_not_nil filter
  842. assert_include 'mine', filter[:values].pluck(1)
  843. query.filters = {'project_id' => {:operator => '=', :values => ['mine']}}
  844. result = query.issues
  845. assert_nil result.detect {|issue| !User.current.member_of?(issue.project)}
  846. end
  847. def test_filter_my_bookmarks
  848. User.current = User.find(1)
  849. query = ProjectQuery.new(:name => '_')
  850. filter = query.available_filters['id']
  851. assert_not_nil filter
  852. assert_include 'bookmarks', filter[:values].pluck(1)
  853. query.filters = {'id' => {:operator => '=', :values => ['bookmarks']}}
  854. result = query.results_scope
  855. assert_equal [1, 5], result.map(&:id).sort
  856. end
  857. def test_filter_my_bookmarks_for_user_without_bookmarked_projects
  858. User.current = User.find(2)
  859. query = ProjectQuery.new(:name => '_')
  860. filter = query.available_filters['id']
  861. assert_not_include 'bookmarks', filter[:values].pluck(1)
  862. end
  863. def test_filter_project_parent_id_with_my_projects
  864. User.current = User.find(1)
  865. query = ProjectQuery.new(:name => '_')
  866. filter = query.available_filters['parent_id']
  867. assert_not_nil filter
  868. assert_include 'mine', filter[:values].pluck(1)
  869. query.filters = {'parent_id' => {:operator => '=', :values => ['mine']}}
  870. result = query.results_scope
  871. my_projects = User.current.memberships.map(&:project_id)
  872. assert_equal Project.where(parent_id: my_projects).ids, result.map(&:id).sort
  873. end
  874. def test_filter_project_parent_id_with_my_bookmarks
  875. User.current = User.find(1)
  876. query = ProjectQuery.new(:name => '_')
  877. filter = query.available_filters['parent_id']
  878. assert_not_nil filter
  879. assert_include 'bookmarks', filter[:values].pluck(1)
  880. query.filters = {'parent_id' => {:operator => '=', :values => ['bookmarks']}}
  881. result = query.results_scope
  882. bookmarks = User.current.bookmarked_project_ids
  883. assert_equal Project.where(parent_id: bookmarks).ids, result.map(&:id).sort
  884. end
  885. def test_filter_watched_issues
  886. User.current = User.find(1)
  887. query =
  888. IssueQuery.new(
  889. :name => '_',
  890. :filters => {
  891. 'watcher_id' => {
  892. :operator => '=',
  893. :values => ['me']
  894. }
  895. }
  896. )
  897. result = find_issues_with_query(query)
  898. assert_not_nil result
  899. assert !result.empty?
  900. assert_equal Issue.visible.watched_by(User.current).sort_by(&:id), result.sort_by(&:id)
  901. end
  902. def test_filter_watched_issues_with_groups_also
  903. user = User.find(2)
  904. group = Group.find(10)
  905. group.users << user
  906. Issue.find(3).add_watcher(user)
  907. Issue.find(7).add_watcher(group)
  908. User.current = user
  909. query =
  910. IssueQuery.new(
  911. :name => '_',
  912. :filters => {
  913. 'watcher_id' => {
  914. :operator => '=',
  915. :values => ['me']
  916. }
  917. }
  918. )
  919. result = find_issues_with_query(query)
  920. assert_not_nil result
  921. assert !result.empty?
  922. assert_equal [3, 7], result.sort_by(&:id).pluck(:id)
  923. end
  924. def test_filter_unwatched_issues
  925. User.current = User.find(1)
  926. query =
  927. IssueQuery.new(
  928. :name => '_',
  929. :filters => {
  930. 'watcher_id' => {
  931. :operator => '!', :values => ['me']
  932. }
  933. }
  934. )
  935. result = find_issues_with_query(query)
  936. assert_not_nil result
  937. assert !result.empty?
  938. assert_equal((Issue.visible - Issue.watched_by(User.current)).sort_by(&:id).size, result.sort_by(&:id).size)
  939. end
  940. def test_filter_on_watched_issues_with_view_issue_watchers_permission
  941. User.current = User.find(1)
  942. User.current.admin = true
  943. assert User.current.allowed_to?(:view_issue_watchers, Project.find(1))
  944. Issue.find(1).add_watcher User.current
  945. Issue.find(3).add_watcher User.find(3)
  946. query =
  947. IssueQuery.new(
  948. :name => '_',
  949. :filters => {
  950. 'watcher_id' => {
  951. :operator => '=',
  952. :values => ['me', '3']
  953. }
  954. }
  955. )
  956. result = find_issues_with_query(query)
  957. assert_includes result, Issue.find(1)
  958. assert_includes result, Issue.find(3)
  959. end
  960. def test_filter_on_watched_issues_without_view_issue_watchers_permission
  961. User.current = User.find(1)
  962. User.current.admin = false
  963. assert !User.current.allowed_to?(:view_issue_watchers, Project.find(1))
  964. Issue.find(1).add_watcher User.current
  965. Issue.find(3).add_watcher User.find(3)
  966. query =
  967. IssueQuery.new(
  968. :name => '_',
  969. :filters => {
  970. 'watcher_id' => {
  971. :operator => '=',
  972. :values => ['me', '3']
  973. }
  974. }
  975. )
  976. result = find_issues_with_query(query)
  977. assert_includes result, Issue.find(1)
  978. assert_not_includes result, Issue.find(3)
  979. end
  980. def test_filter_on_custom_field_should_ignore_projects_with_field_disabled
  981. field =
  982. IssueCustomField.generate!(
  983. :trackers => Tracker.all, :project_ids => [1, 3, 5],
  984. :is_for_all => false, :is_filter => true
  985. )
  986. Issue.generate!(:project_id => 3, :tracker_id => 2,
  987. :custom_field_values => {field.id.to_s => 'Foo'})
  988. Issue.generate!(:project_id => 5, :tracker_id => 2,
  989. :custom_field_values => {field.id.to_s => 'Foo'})
  990. User.current = User.find(1)
  991. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  992. query.filters = {"cf_#{field.id}" => {:operator => '=', :values => ['Foo']}}
  993. assert_equal 2, find_issues_with_query(query).size
  994. field.project_ids = [1, 3] # Disable the field for project 4
  995. field.save!
  996. assert_equal 1, find_issues_with_query(query).size
  997. end
  998. def test_filter_on_custom_field_should_ignore_trackers_with_field_disabled
  999. field = IssueCustomField.generate!(:tracker_ids => [1, 2], :is_for_all => true, :is_filter => true)
  1000. Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {field.id.to_s => 'Foo'})
  1001. Issue.generate!(:project_id => 1, :tracker_id => 2, :custom_field_values => {field.id.to_s => 'Foo'})
  1002. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  1003. query.filters = {"cf_#{field.id}" => {:operator => '=', :values => ['Foo']}}
  1004. assert_equal 2, find_issues_with_query(query).size
  1005. field.tracker_ids = [1] # Disable the field for tracker 2
  1006. field.save!
  1007. assert_equal 1, find_issues_with_query(query).size
  1008. end
  1009. def test_filter_on_project_custom_field
  1010. field = ProjectCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
  1011. CustomValue.create!(:custom_field => field, :customized => Project.find(3), :value => 'Foo')
  1012. CustomValue.create!(:custom_field => field, :customized => Project.find(5), :value => 'Foo')
  1013. query = IssueQuery.new(:name => '_')
  1014. filter_name = "project.cf_#{field.id}"
  1015. assert_include filter_name, query.available_filters.keys
  1016. query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
  1017. assert_equal [3, 5], find_issues_with_query(query).map(&:project_id).uniq.sort
  1018. end
  1019. def test_filter_on_author_custom_field
  1020. field = UserCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
  1021. CustomValue.create!(:custom_field => field, :customized => User.find(3), :value => 'Foo')
  1022. query = IssueQuery.new(:name => '_')
  1023. filter_name = "author.cf_#{field.id}"
  1024. assert_include filter_name, query.available_filters.keys
  1025. query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
  1026. assert_equal [3], find_issues_with_query(query).map(&:author_id).uniq.sort
  1027. end
  1028. def test_filter_on_assigned_to_custom_field
  1029. field = UserCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
  1030. CustomValue.create!(:custom_field => field, :customized => User.find(3), :value => 'Foo')
  1031. query = IssueQuery.new(:name => '_')
  1032. filter_name = "assigned_to.cf_#{field.id}"
  1033. assert_include filter_name, query.available_filters.keys
  1034. query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
  1035. assert_equal [3], find_issues_with_query(query).map(&:assigned_to_id).uniq.sort
  1036. end
  1037. def test_filter_on_fixed_version_custom_field
  1038. field = VersionCustomField.create!(:name => 'Client', :is_filter => true, :field_format => 'string')
  1039. CustomValue.create!(:custom_field => field, :customized => Version.find(2), :value => 'Foo')
  1040. query = IssueQuery.new(:name => '_')
  1041. filter_name = "fixed_version.cf_#{field.id}"
  1042. assert_include filter_name, query.available_filters.keys
  1043. query.filters = {filter_name => {:operator => '=', :values => ['Foo']}}
  1044. assert_equal [2], find_issues_with_query(query).map(&:fixed_version_id).uniq.sort
  1045. end
  1046. def test_filter_on_fixed_version_due_date
  1047. query = IssueQuery.new(:name => '_')
  1048. filter_name = "fixed_version.due_date"
  1049. assert_include filter_name, query.available_filters.keys
  1050. query.filters = {filter_name => {:operator => '=', :values => [20.day.from_now.to_date.to_s(:db)]}}
  1051. issues = find_issues_with_query(query)
  1052. assert_equal [2], issues.map(&:fixed_version_id).uniq.sort
  1053. assert_equal [2, 12], issues.map(&:id).sort
  1054. query = IssueQuery.new(:name => '_')
  1055. query.filters = {filter_name => {:operator => '>=', :values => [21.day.from_now.to_date.to_s(:db)]}}
  1056. assert_equal 0, find_issues_with_query(query).size
  1057. end
  1058. def test_filter_on_fixed_version_status
  1059. query = IssueQuery.new(:name => '_')
  1060. filter_name = "fixed_version.status"
  1061. assert_include filter_name, query.available_filters.keys
  1062. query.filters = {filter_name => {:operator => '=', :values => ['closed']}}
  1063. issues = find_issues_with_query(query)
  1064. assert_equal [1], issues.map(&:fixed_version_id).sort
  1065. assert_equal [11], issues.map(&:id).sort
  1066. # "is not" operator should include issues without target version
  1067. query = IssueQuery.new(:name => '_')
  1068. query.filters = {filter_name => {:operator => '!', :values => ['open', 'closed', 'locked']}, "project_id" => {:operator => '=', :values => [1]}}
  1069. assert_equal [1, 3, 7, 8], find_issues_with_query(query).map(&:id).uniq.sort
  1070. end
  1071. def test_filter_on_fixed_version_status_respects_sharing
  1072. issue = Issue.generate!(:project_id => 1, :fixed_version_id => 7)
  1073. filter_name = "fixed_version.status"
  1074. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  1075. assert_include filter_name, query.available_filters.keys
  1076. query.filters = {filter_name => {:operator => '=', :values => ['open']}}
  1077. assert_include issue, find_issues_with_query(query)
  1078. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  1079. query.filters = {filter_name => {:operator => '=', :values => ['closed']}}
  1080. assert_not_includes find_issues_with_query(query), issue
  1081. end
  1082. def test_filter_on_version_custom_field
  1083. field = IssueCustomField.generate!(:field_format => 'version', :is_filter => true)
  1084. issue = Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {field.id.to_s => '2'})
  1085. query = IssueQuery.new(:name => '_')
  1086. filter_name = "cf_#{field.id}"
  1087. assert_include filter_name, query.available_filters.keys
  1088. query.filters = {filter_name => {:operator => '=', :values => ['2']}}
  1089. issues = find_issues_with_query(query)
  1090. assert_equal [issue.id], issues.map(&:id).sort
  1091. end
  1092. def test_filter_on_attribute_of_version_custom_field
  1093. field = IssueCustomField.generate!(:field_format => 'version', :is_filter => true)
  1094. version = Version.generate!(:effective_date => '2017-01-14')
  1095. issue = Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {field.id.to_s => version.id.to_s})
  1096. query = IssueQuery.new(:name => '_')
  1097. filter_name = "cf_#{field.id}.due_date"
  1098. assert_include filter_name, query.available_filters.keys
  1099. query.filters = {filter_name => {:operator => '=', :values => ['2017-01-14']}}
  1100. issues = find_issues_with_query(query)
  1101. assert_equal [issue.id], issues.map(&:id).sort
  1102. end
  1103. def test_filter_on_custom_field_of_version_custom_field
  1104. field = IssueCustomField.generate!(:field_format => 'version', :is_filter => true)
  1105. attr = VersionCustomField.generate!(:field_format => 'string', :is_filter => true)
  1106. version = Version.generate!(:custom_field_values => {attr.id.to_s => 'ABC'})
  1107. issue = Issue.generate!(:project_id => 1, :tracker_id => 1, :custom_field_values => {field.id.to_s => version.id.to_s})
  1108. query = IssueQuery.new(:name => '_')
  1109. filter_name = "cf_#{field.id}.cf_#{attr.id}"
  1110. assert_include filter_name, query.available_filters.keys
  1111. query.filters = {filter_name => {:operator => '=', :values => ['ABC']}}
  1112. issues = find_issues_with_query(query)
  1113. assert_equal [issue.id], issues.map(&:id).sort
  1114. end
  1115. def test_filter_on_relations_with_a_specific_issue
  1116. IssueRelation.delete_all
  1117. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
  1118. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
  1119. query = IssueQuery.new(:name => '_')
  1120. query.filters = {"relates" => {:operator => '=', :values => ['1']}}
  1121. assert_equal [2, 3], find_issues_with_query(query).map(&:id).sort
  1122. query = IssueQuery.new(:name => '_')
  1123. query.filters = {"relates" => {:operator => '=', :values => ['2']}}
  1124. assert_equal [1], find_issues_with_query(query).map(&:id).sort
  1125. query = IssueQuery.new(:name => '_')
  1126. query.filters = {"relates" => {:operator => '!', :values => ['1']}}
  1127. assert_equal Issue.where.not(:id => [2, 3]).order(:id).ids, find_issues_with_query(query).map(&:id).sort
  1128. end
  1129. def test_filter_on_relations_with_any_issues_in_a_project
  1130. IssueRelation.delete_all
  1131. with_settings :cross_project_issue_relations => '1' do
  1132. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
  1133. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(2).issues.first)
  1134. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(3).issues.first)
  1135. end
  1136. query = IssueQuery.new(:name => '_')
  1137. query.filters = {"relates" => {:operator => '=p', :values => ['2']}}
  1138. assert_equal [1, 2], find_issues_with_query(query).map(&:id).sort
  1139. query = IssueQuery.new(:name => '_')
  1140. query.filters = {"relates" => {:operator => '=p', :values => ['3']}}
  1141. assert_equal [1], find_issues_with_query(query).map(&:id).sort
  1142. query = IssueQuery.new(:name => '_')
  1143. query.filters = {"relates" => {:operator => '=p', :values => ['4']}}
  1144. assert_equal [], find_issues_with_query(query).map(&:id).sort
  1145. end
  1146. def test_filter_on_relations_with_any_issues_not_in_a_project
  1147. IssueRelation.delete_all
  1148. with_settings :cross_project_issue_relations => '1' do
  1149. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
  1150. # IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(1).issues.first)
  1151. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(3).issues.first)
  1152. end
  1153. query = IssueQuery.new(:name => '_')
  1154. query.filters = {"relates" => {:operator => '=!p', :values => ['1']}}
  1155. assert_equal [1], find_issues_with_query(query).map(&:id).sort
  1156. end
  1157. def test_filter_on_relations_with_no_issues_in_a_project
  1158. IssueRelation.delete_all
  1159. with_settings :cross_project_issue_relations => '1' do
  1160. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Project.find(2).issues.first)
  1161. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(2), :issue_to => Project.find(3).issues.first)
  1162. IssueRelation.create!(:relation_type => "relates", :issue_to => Project.find(2).issues.first, :issue_from => Issue.find(3))
  1163. end
  1164. query = IssueQuery.new(:name => '_')
  1165. query.filters = {"relates" => {:operator => '!p', :values => ['2']}}
  1166. ids = find_issues_with_query(query).map(&:id).sort
  1167. assert_include 2, ids
  1168. assert_not_include 1, ids
  1169. assert_not_include 3, ids
  1170. end
  1171. def test_filter_on_relations_with_any_open_issues
  1172. IssueRelation.delete_all
  1173. # Issue 1 is blocked by 8, which is closed
  1174. IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(1), :issue_to => Issue.find(8))
  1175. # Issue 2 is blocked by 3, which is open
  1176. IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(2), :issue_to => Issue.find(3))
  1177. query = IssueQuery.new(:name => '_')
  1178. query.filters = {"blocked" => {:operator => "*o", :values => ['']}}
  1179. ids = find_issues_with_query(query).map(&:id)
  1180. assert_equal [], ids & [1]
  1181. assert_include 2, ids
  1182. end
  1183. def test_filter_on_blocked_by_no_open_issues
  1184. IssueRelation.delete_all
  1185. # Issue 1 is blocked by 8, which is closed
  1186. IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(1), :issue_to => Issue.find(8))
  1187. # Issue 2 is blocked by 3, which is open
  1188. IssueRelation.create!(:relation_type => "blocked", :issue_from => Issue.find(2), :issue_to => Issue.find(3))
  1189. query = IssueQuery.new(:name => '_')
  1190. query.filters = {"blocked" => {:operator => "!o", :values => ['']}}
  1191. ids = find_issues_with_query(query).map(&:id)
  1192. assert_equal [], ids & [2]
  1193. assert_include 1, ids
  1194. end
  1195. def test_filter_on_related_with_no_open_issues
  1196. IssueRelation.delete_all
  1197. # Issue 1 is blocked by 8, which is closed
  1198. IssueRelation.create!(relation_type: 'relates', issue_from: Issue.find(1), issue_to: Issue.find(8))
  1199. # Issue 2 is blocked by 3, which is open
  1200. IssueRelation.create!(relation_type: 'relates', issue_from: Issue.find(2), issue_to: Issue.find(3))
  1201. query = IssueQuery.new(:name => '_')
  1202. query.filters = {'relates' => {:operator => '!o', :values => ['']}}
  1203. ids = find_issues_with_query(query).map(&:id)
  1204. assert_equal [], ids & [2]
  1205. assert_include 1, ids
  1206. end
  1207. def test_filter_on_relations_with_no_issues
  1208. IssueRelation.delete_all
  1209. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
  1210. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
  1211. query = IssueQuery.new(:name => '_')
  1212. query.filters = {"relates" => {:operator => '!*', :values => ['']}}
  1213. ids = find_issues_with_query(query).map(&:id)
  1214. assert_equal [], ids & [1, 2, 3]
  1215. assert_include 4, ids
  1216. end
  1217. def test_filter_on_relations_with_any_issues
  1218. IssueRelation.delete_all
  1219. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(1), :issue_to => Issue.find(2))
  1220. IssueRelation.create!(:relation_type => "relates", :issue_from => Issue.find(3), :issue_to => Issue.find(1))
  1221. query = IssueQuery.new(:name => '_')
  1222. query.filters = {"relates" => {:operator => '*', :values => ['']}}
  1223. assert_equal [1, 2, 3], find_issues_with_query(query).map(&:id).sort
  1224. end
  1225. def test_filter_on_relations_should_not_ignore_other_filter
  1226. issue = Issue.generate!
  1227. issue1 = Issue.generate!(:status_id => 1)
  1228. issue2 = Issue.generate!(:status_id => 2)
  1229. IssueRelation.create!(:relation_type => "relates", :issue_from => issue, :issue_to => issue1)
  1230. IssueRelation.create!(:relation_type => "relates", :issue_from => issue, :issue_to => issue2)
  1231. query = IssueQuery.new(:name => '_')
  1232. query.filters = {
  1233. "status_id" => {:operator => '=', :values => ['1']},
  1234. "relates" => {:operator => '=', :values => [issue.id.to_s]}
  1235. }
  1236. assert_equal [issue1], find_issues_with_query(query)
  1237. end
  1238. def test_filter_on_parent
  1239. Issue.delete_all
  1240. parent = Issue.generate_with_descendants!
  1241. query = IssueQuery.new(:name => '_')
  1242. query.filters = {"parent_id" => {:operator => '=', :values => [parent.id.to_s]}}
  1243. assert_equal parent.children.map(&:id).sort, find_issues_with_query(query).map(&:id).sort
  1244. query.filters = {"parent_id" => {:operator => '~', :values => [parent.id.to_s]}}
  1245. assert_equal parent.descendants.map(&:id).sort, find_issues_with_query(query).map(&:id).sort
  1246. query.filters = {"parent_id" => {:operator => '*', :values => ['']}}
  1247. assert_equal parent.descendants.map(&:id).sort, find_issues_with_query(query).map(&:id).sort
  1248. query.filters = {"parent_id" => {:operator => '!*', :values => ['']}}
  1249. assert_equal [parent.id], find_issues_with_query(query).map(&:id).sort
  1250. end
  1251. def test_filter_on_invalid_parent_should_return_no_results
  1252. query = IssueQuery.new(:name => '_')
  1253. query.filters = {"parent_id" => {:operator => '=', :values => '99999999999'}}
  1254. assert_equal [], find_issues_with_query(query).map(&:id).sort
  1255. query.filters = {"parent_id" => {:operator => '~', :values => '99999999999'}}
  1256. assert_equal [], find_issues_with_query(query)
  1257. end
  1258. def test_filter_on_child
  1259. Issue.delete_all
  1260. parent = Issue.generate_with_descendants!
  1261. child, leaf = parent.children.sort_by(&:id)
  1262. grandchild = child.children.first
  1263. query = IssueQuery.new(:name => '_')
  1264. query.filters = {"child_id" => {:operator => '=', :values => [grandchild.id.to_s]}}
  1265. assert_equal [child.id], find_issues_with_query(query).map(&:id).sort
  1266. query.filters = {"child_id" => {:operator => '~', :values => [grandchild.id.to_s]}}
  1267. assert_equal [parent, child].map(&:id).sort, find_issues_with_query(query).map(&:id).sort
  1268. query.filters = {"child_id" => {:operator => '*', :values => ['']}}
  1269. assert_equal [parent, child].map(&:id).sort, find_issues_with_query(query).map(&:id).sort
  1270. query.filters = {"child_id" => {:operator => '!*', :values => ['']}}
  1271. assert_equal [grandchild, leaf].map(&:id).sort, find_issues_with_query(query).map(&:id).sort
  1272. end
  1273. def test_filter_on_invalid_child_should_return_no_results
  1274. query = IssueQuery.new(:name => '_')
  1275. query.filters = {"child_id" => {:operator => '=', :values => '99999999999'}}
  1276. assert_equal [], find_issues_with_query(query)
  1277. query.filters = {"child_id" => {:operator => '~', :values => '99999999999'}}
  1278. assert_equal [].map(&:id).sort, find_issues_with_query(query)
  1279. end
  1280. def test_filter_on_attachment_any
  1281. query = IssueQuery.new(:name => '_')
  1282. query.filters = {"attachment" => {:operator => '*', :values => ['']}}
  1283. issues = find_issues_with_query(query)
  1284. assert issues.any?
  1285. assert_nil issues.detect {|issue| issue.attachments.empty?}
  1286. end
  1287. def test_filter_on_attachment_none
  1288. query = IssueQuery.new(:name => '_')
  1289. query.filters = {"attachment" => {:operator => '!*', :values => ['']}}
  1290. issues = find_issues_with_query(query)
  1291. assert issues.any?
  1292. assert_nil issues.detect {|issue| issue.attachments.any?}
  1293. end
  1294. def test_filter_on_attachment_contains
  1295. query = IssueQuery.new(:name => '_')
  1296. query.filters = {"attachment" => {:operator => '~', :values => ['error281']}}
  1297. issues = find_issues_with_query(query)
  1298. assert issues.any?
  1299. assert_nil issues.detect {|issue| ! issue.attachments.any? {|attachment| attachment.filename.include?('error281')}}
  1300. end
  1301. def test_filter_on_attachment_not_contains
  1302. query = IssueQuery.new(:name => '_')
  1303. query.filters = {"attachment" => {:operator => '!~', :values => ['error281']}}
  1304. issues = find_issues_with_query(query)
  1305. assert issues.any?
  1306. assert_nil issues.detect {|issue| issue.attachments.any? {|attachment| attachment.filename.include?('error281')}}
  1307. end
  1308. def test_filter_on_attachment_when_starts_with
  1309. query = IssueQuery.new(:name => '_')
  1310. query.filters = {"attachment" => {:operator => '^', :values => ['testfile']}}
  1311. issues = find_issues_with_query(query)
  1312. assert_equal [14], issues.collect(&:id).sort
  1313. end
  1314. def test_filter_on_attachment_when_ends_with
  1315. query = IssueQuery.new(:name => '_')
  1316. query.filters = {"attachment" => {:operator => '$', :values => ['zip']}}
  1317. issues = find_issues_with_query(query)
  1318. assert_equal [3, 4], issues.collect(&:id).sort
  1319. end
  1320. def test_filter_on_attachment_description_when_any
  1321. query = IssueQuery.new(:name => '_')
  1322. query.filters = {"attachment_description" => {:operator => '*', :values => ['']}}
  1323. issues = find_issues_with_query(query)
  1324. assert_equal [2, 3, 14], issues.collect(&:id).sort
  1325. end
  1326. def test_filter_on_attachment_description_when_none
  1327. query = IssueQuery.new(:name => '_')
  1328. query.filters = {"attachment_description" => {:operator => '!*', :values => ['']}}
  1329. issues = find_issues_with_query(query)
  1330. assert_equal [2, 3, 4, 14], issues.collect(&:id).sort
  1331. end
  1332. def test_filter_on_attachment_description_when_contains
  1333. query = IssueQuery.new(:name => '_')
  1334. query.filters = {"attachment_description" => {:operator => '~', :values => ['attachment']}}
  1335. issues = find_issues_with_query(query)
  1336. assert_equal [3, 14], issues.collect(&:id).sort
  1337. end
  1338. def test_filter_on_attachment_description_when_does_not_contain
  1339. query = IssueQuery.new(:name => '_')
  1340. query.filters = {"attachment_description" => {:operator => '!~', :values => ['attachment']}}
  1341. issues = find_issues_with_query(query)
  1342. assert_equal [2], issues.collect(&:id).sort
  1343. end
  1344. def test_filter_on_attachment_description_when_starts_with
  1345. query = IssueQuery.new(:name => '_')
  1346. query.filters = {"attachment_description" => {:operator => '^', :values => ['attachment']}}
  1347. issues = find_issues_with_query(query)
  1348. assert_equal [14], issues.collect(&:id).sort
  1349. end
  1350. def test_filter_on_attachment_description_when_ends_with
  1351. query = IssueQuery.new(:name => '_')
  1352. query.filters = {"attachment_description" => {:operator => '$', :values => ['attachment']}}
  1353. issues = find_issues_with_query(query)
  1354. assert_equal [3], issues.collect(&:id).sort
  1355. end
  1356. def test_filter_on_subject_when_starts_with
  1357. query = IssueQuery.new(:name => '_')
  1358. query.filters = {'subject' => {:operator => '^', :values => ['issue']}}
  1359. issues = find_issues_with_query(query)
  1360. assert_equal [4, 6, 7, 10], issues.collect(&:id).sort
  1361. end
  1362. def test_filter_on_subject_when_ends_with
  1363. query = IssueQuery.new(:name => '_')
  1364. query.filters = {'subject' => {:operator => '$', :values => ['issue']}}
  1365. issues = find_issues_with_query(query)
  1366. assert_equal [5, 8, 9], issues.collect(&:id).sort
  1367. end
  1368. def test_statement_should_be_nil_with_no_filters
  1369. q = IssueQuery.new(:name => '_')
  1370. q.filters = {}
  1371. assert q.valid?
  1372. assert_nil q.statement
  1373. end
  1374. def test_available_filters_as_json_should_include_missing_assigned_to_id_values
  1375. user = User.generate!
  1376. with_current_user User.find(1) do
  1377. q = IssueQuery.new
  1378. q.filters = {"assigned_to_id" => {:operator => '=', :values => user.id.to_s}}
  1379. filters = q.available_filters_as_json
  1380. assert_include [user.name, user.id.to_s], filters['assigned_to_id']['values']
  1381. end
  1382. end
  1383. def test_available_filters_as_json_should_not_include_duplicate_assigned_to_id_values
  1384. set_language_if_valid 'en'
  1385. user = User.find_by_login 'dlopper'
  1386. with_current_user User.find(1) do
  1387. q = IssueQuery.new
  1388. q.filters = {"assigned_to_id" => {:operator => '=', :values => user.id.to_s}}
  1389. filters = q.available_filters_as_json
  1390. assert_not_include [user.name, user.id.to_s], filters['assigned_to_id']['values']
  1391. assert_include [user.name, user.id.to_s, 'active'], filters['assigned_to_id']['values']
  1392. end
  1393. end
  1394. def test_available_filters_as_json_should_include_missing_author_id_values
  1395. user = User.generate!
  1396. with_current_user User.find(1) do
  1397. q = IssueQuery.new
  1398. q.filters = {"author_id" => {:operator => '=', :values => user.id.to_s}}
  1399. filters = q.available_filters_as_json
  1400. assert_include [user.name, user.id.to_s], filters['author_id']['values']
  1401. end
  1402. end
  1403. def test_default_columns
  1404. q = IssueQuery.new
  1405. assert q.columns.any?
  1406. assert q.inline_columns.any?
  1407. assert q.block_columns.empty?
  1408. end
  1409. def test_set_column_names
  1410. q = IssueQuery.new
  1411. q.column_names = ['tracker', :subject, '', 'unknonw_column']
  1412. assert_equal [:id, :tracker, :subject], q.columns.collect {|c| c.name}
  1413. end
  1414. def test_has_column_should_accept_a_column_name
  1415. q = IssueQuery.new
  1416. q.column_names = ['tracker', :subject]
  1417. assert q.has_column?(:tracker)
  1418. assert !q.has_column?(:category)
  1419. end
  1420. def test_has_column_should_accept_a_column
  1421. q = IssueQuery.new
  1422. q.column_names = ['tracker', :subject]
  1423. tracker_column = q.available_columns.detect {|c| c.name==:tracker}
  1424. assert_kind_of QueryColumn, tracker_column
  1425. category_column = q.available_columns.detect {|c| c.name==:category}
  1426. assert_kind_of QueryColumn, category_column
  1427. assert q.has_column?(tracker_column)
  1428. assert !q.has_column?(category_column)
  1429. end
  1430. def test_has_column_should_return_true_for_default_column
  1431. with_settings :issue_list_default_columns => %w(tracker subject) do
  1432. q = IssueQuery.new
  1433. assert q.has_column?(:tracker)
  1434. assert !q.has_column?(:category)
  1435. end
  1436. end
  1437. def test_inline_and_block_columns
  1438. q = IssueQuery.new
  1439. q.column_names = ['subject', 'description', 'tracker', 'last_notes']
  1440. assert_equal [:id, :subject, :tracker], q.inline_columns.map(&:name)
  1441. assert_equal [:description, :last_notes], q.block_columns.map(&:name)
  1442. end
  1443. def test_custom_field_columns_should_be_inline
  1444. q = IssueQuery.new
  1445. columns = q.available_columns.select {|column| column.is_a? QueryCustomFieldColumn}
  1446. assert columns.any?
  1447. assert_nil columns.detect {|column| !column.inline?}
  1448. end
  1449. def test_query_should_preload_spent_hours
  1450. q = IssueQuery.new(:name => '_', :column_names => [:subject, :spent_hours])
  1451. assert q.has_column?(:spent_hours)
  1452. issues = q.issues
  1453. assert_not_nil issues.first.instance_variable_get(:@spent_hours)
  1454. end
  1455. def test_query_should_preload_last_updated_by
  1456. with_current_user User.find(2) do
  1457. q = IssueQuery.new(:name => '_', :column_names => [:subject, :last_updated_by])
  1458. q.filters = {"issue_id" => {:operator => '=', :values => ['1,2,3']}}
  1459. assert q.has_column?(:last_updated_by)
  1460. issues = q.issues.sort_by(&:id)
  1461. assert issues.all? {|issue| !issue.instance_variable_get(:@last_updated_by).nil?}
  1462. assert_equal ["User", "User", "NilClass"], issues.map {|i| i.last_updated_by.class.name}
  1463. assert_equal ["John Smith", "John Smith", ""], issues.map {|i| i.last_updated_by.to_s}
  1464. end
  1465. end
  1466. def test_query_should_preload_last_notes
  1467. q = IssueQuery.new(:name => '_', :column_names => [:subject, :last_notes])
  1468. assert q.has_column?(:last_notes)
  1469. issues = q.issues
  1470. assert_not_nil issues.first.instance_variable_get(:@last_notes)
  1471. end
  1472. def test_groupable_columns_should_include_custom_fields
  1473. q = IssueQuery.new
  1474. column = q.groupable_columns.detect {|c| c.name == :cf_1}
  1475. assert_not_nil column
  1476. assert_kind_of QueryCustomFieldColumn, column
  1477. end
  1478. def test_groupable_columns_should_not_include_multi_custom_fields
  1479. field = CustomField.find(1)
  1480. field.update_attribute :multiple, true
  1481. q = IssueQuery.new
  1482. column = q.groupable_columns.detect {|c| c.name == :cf_1}
  1483. assert_nil column
  1484. end
  1485. def test_groupable_columns_should_include_user_custom_fields
  1486. cf =
  1487. IssueCustomField.create!(
  1488. :name => 'User', :is_for_all => true, :tracker_ids => [1],
  1489. :field_format => 'user'
  1490. )
  1491. q = IssueQuery.new
  1492. assert q.groupable_columns.detect {|c| c.name == "cf_#{cf.id}".to_sym}
  1493. end
  1494. def test_groupable_columns_should_include_version_custom_fields
  1495. cf =
  1496. IssueCustomField.create!(
  1497. :name => 'User', :is_for_all => true,
  1498. :tracker_ids => [1], :field_format => 'version'
  1499. )
  1500. q = IssueQuery.new
  1501. assert q.groupable_columns.detect {|c| c.name == "cf_#{cf.id}".to_sym}
  1502. end
  1503. def test_grouped_with_valid_column
  1504. q = IssueQuery.new(:group_by => 'status')
  1505. assert q.grouped?
  1506. assert_not_nil q.group_by_column
  1507. assert_equal :status, q.group_by_column.name
  1508. assert_not_nil q.group_by_statement
  1509. assert_equal 'status', q.group_by_statement
  1510. end
  1511. def test_grouped_with_invalid_column
  1512. q = IssueQuery.new(:group_by => 'foo')
  1513. assert !q.grouped?
  1514. assert_nil q.group_by_column
  1515. assert_nil q.group_by_statement
  1516. end
  1517. def test_sortable_columns_should_sort_assignees_according_to_user_format_setting
  1518. with_settings :user_format => 'lastname_comma_firstname' do
  1519. q = IssueQuery.new
  1520. assert q.sortable_columns.has_key?('assigned_to')
  1521. assert_equal %w(users.lastname users.firstname users.id), q.sortable_columns['assigned_to']
  1522. end
  1523. end
  1524. def test_sortable_columns_should_sort_authors_according_to_user_format_setting
  1525. with_settings :user_format => 'lastname_comma_firstname' do
  1526. q = IssueQuery.new
  1527. assert q.sortable_columns.has_key?('author')
  1528. assert_equal %w(authors.lastname authors.firstname authors.id), q.sortable_columns['author']
  1529. end
  1530. end
  1531. def test_sortable_columns_should_sort_last_updated_by_according_to_user_format_setting
  1532. with_settings :user_format => 'lastname_comma_firstname' do
  1533. q = IssueQuery.new
  1534. q.sort_criteria = [['last_updated_by', 'desc']]
  1535. assert q.sortable_columns.has_key?('last_updated_by')
  1536. assert_equal(
  1537. %w(last_journal_user.lastname last_journal_user.firstname last_journal_user.id),
  1538. q.sortable_columns['last_updated_by']
  1539. )
  1540. end
  1541. end
  1542. def test_sortable_columns_should_include_custom_field
  1543. q = IssueQuery.new
  1544. assert q.sortable_columns['cf_1']
  1545. end
  1546. def test_sortable_columns_should_not_include_multi_custom_field
  1547. field = CustomField.find(1)
  1548. field.update_attribute :multiple, true
  1549. q = IssueQuery.new
  1550. assert !q.sortable_columns['cf_1']
  1551. end
  1552. def test_sortable_should_return_false_for_multi_custom_field
  1553. field = CustomField.find(1)
  1554. field.update_attribute :multiple, true
  1555. q = IssueQuery.new
  1556. field_column = q.available_columns.detect {|c| c.name==:cf_1}
  1557. assert !field_column.sortable?
  1558. end
  1559. def test_default_sort
  1560. q = IssueQuery.new
  1561. assert_equal [['id', 'desc']], q.sort_criteria
  1562. end
  1563. def test_sort_criteria_should_have_only_first_three_elements
  1564. q = IssueQuery.new
  1565. q.sort_criteria = [
  1566. ['priority', 'desc'], ['tracker', 'asc'], ['priority', 'asc'],
  1567. ['id', 'asc'], ['project', 'asc'], ['subject', 'asc']
  1568. ]
  1569. assert_equal [['priority', 'desc'], ['tracker', 'asc'], ['id', 'asc']], q.sort_criteria
  1570. end
  1571. def test_sort_criteria_should_remove_blank_or_duplicate_keys
  1572. q = IssueQuery.new
  1573. q.sort_criteria = [['priority', 'desc'], [nil, 'desc'], ['', 'asc'], ['priority', 'asc'], ['project', 'asc']]
  1574. assert_equal [['priority', 'desc'], ['project', 'asc']], q.sort_criteria
  1575. end
  1576. def test_set_sort_criteria_with_hash
  1577. q = IssueQuery.new
  1578. q.sort_criteria = {'0' => ['priority', 'desc'], '2' => ['tracker']}
  1579. assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
  1580. end
  1581. def test_set_sort_criteria_with_array
  1582. q = IssueQuery.new
  1583. q.sort_criteria = [['priority', 'desc'], 'tracker']
  1584. assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
  1585. end
  1586. def test_create_query_with_sort
  1587. q = IssueQuery.new(:name => 'Sorted')
  1588. q.sort_criteria = [['priority', 'desc'], 'tracker']
  1589. assert q.save
  1590. q.reload
  1591. assert_equal [['priority', 'desc'], ['tracker', 'asc']], q.sort_criteria
  1592. end
  1593. def test_sort_by_string_custom_field_asc
  1594. q = IssueQuery.new
  1595. c =
  1596. q.available_columns.find do |col|
  1597. col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'string'
  1598. end
  1599. assert c
  1600. assert c.sortable
  1601. q.sort_criteria = [[c.name.to_s, 'asc']]
  1602. issues = q.issues
  1603. values = issues.collect {|i| i.custom_value_for(c.custom_field).to_s}
  1604. assert !values.empty?
  1605. assert_equal values.sort, values
  1606. end
  1607. def test_sort_by_string_custom_field_desc
  1608. q = IssueQuery.new
  1609. c =
  1610. q.available_columns.find do |col|
  1611. col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'string'
  1612. end
  1613. assert c
  1614. assert c.sortable
  1615. q.sort_criteria = [[c.name.to_s, 'desc']]
  1616. issues = q.issues
  1617. values = issues.collect {|i| i.custom_value_for(c.custom_field).to_s}
  1618. assert !values.empty?
  1619. assert_equal values.sort.reverse, values
  1620. end
  1621. def test_sort_by_float_custom_field_asc
  1622. q = IssueQuery.new
  1623. c =
  1624. q.available_columns.find do |col|
  1625. col.is_a?(QueryCustomFieldColumn) && col.custom_field.field_format == 'float'
  1626. end
  1627. assert c
  1628. assert c.sortable
  1629. q.sort_criteria = [[c.name.to_s, 'asc']]
  1630. issues = q.issues
  1631. values =
  1632. issues.filter_map do |i|
  1633. begin
  1634. Kernel.Float(i.custom_value_for(c.custom_field).to_s)
  1635. rescue
  1636. nil
  1637. end
  1638. end
  1639. assert !values.empty?
  1640. assert_equal values.sort, values
  1641. end
  1642. def test_sort_with_group_by_timestamp_query_column_should_sort_after_date_value
  1643. User.current = User.find(1)
  1644. # Touch Issue#10 in order to be the last updated issue
  1645. Issue.find(10).update_attribute(:updated_on, Issue.find(10).updated_on + 1)
  1646. q = IssueQuery.new(
  1647. :name => '_',
  1648. :filters => {'updated_on' => {:operator => 't', :values => ['']}},
  1649. :group_by => 'updated_on',
  1650. :sort_criteria => [['subject', 'asc']]
  1651. )
  1652. # The following 3 issues are updated today (ordered by updated_on):
  1653. # Issue#10: Issue Doing the Blocking
  1654. # Issue#9: Blocked Issue
  1655. # Issue#6: Issue of a private subproject
  1656. # When we group by a timestamp query column, all the issues in the group have the same date value (today)
  1657. # and the time of the value should not be taken into consideration when sorting
  1658. #
  1659. # For the same issues after subject ascending should return the following:
  1660. # Issue#9: Blocked Issue
  1661. # Issue#10: Issue Doing the Blocking
  1662. # Issue#6: Issue of a private subproject
  1663. assert_equal [9, 10, 6], q.issues.map(&:id)
  1664. end
  1665. def test_sort_by_total_for_estimated_hours
  1666. # Prepare issues
  1667. parent = issues(:issues_001)
  1668. child = issues(:issues_002)
  1669. private_child = issues(:issues_003)
  1670. other = issues(:issues_007)
  1671. User.current = users(:users_001)
  1672. parent.safe_attributes = {:estimated_hours => 1}
  1673. child.safe_attributes = {:estimated_hours => 2, :parent_issue_id => 1}
  1674. private_child.safe_attributes = {:estimated_hours => 4, :parent_issue_id => 1, :is_private => true}
  1675. other.safe_attributes = {:estimated_hours => 5}
  1676. [parent, child, private_child, other].each(&:save!)
  1677. q = IssueQuery.new(
  1678. :name => '_',
  1679. :filters => {'issue_id' => {:operator => '=', :values => ['1,7']}},
  1680. :sort_criteria => [['total_estimated_hours', 'asc']]
  1681. )
  1682. # With private_child, `parent' is "bigger" than `other'
  1683. ids = q.issue_ids
  1684. assert_equal [7, 1], ids, "Private issue was not used to calculate sort order"
  1685. # Without the invisible private_child, `other' is "bigger" than `parent'
  1686. User.current = User.anonymous
  1687. ids = q.issue_ids
  1688. assert_equal [1, 7], ids, "Private issue was used to calculate sort order"
  1689. end
  1690. def test_set_totalable_names
  1691. q = IssueQuery.new
  1692. q.totalable_names = ['estimated_hours', :spent_hours, '']
  1693. assert_equal [:estimated_hours, :spent_hours], q.totalable_columns.map(&:name)
  1694. end
  1695. def test_totalable_columns_should_default_to_settings
  1696. with_settings :issue_list_default_totals => ['estimated_hours'] do
  1697. q = IssueQuery.new
  1698. assert_equal [:estimated_hours], q.totalable_columns.map(&:name)
  1699. end
  1700. end
  1701. def test_available_totalable_columns_should_include_estimated_hours
  1702. q = IssueQuery.new
  1703. assert_include :estimated_hours, q.available_totalable_columns.map(&:name)
  1704. end
  1705. def test_available_totalable_columns_should_include_spent_hours
  1706. User.current = User.find(1)
  1707. q = IssueQuery.new
  1708. assert_include :spent_hours, q.available_totalable_columns.map(&:name)
  1709. end
  1710. def test_available_totalable_columns_should_include_int_custom_field
  1711. field = IssueCustomField.generate!(:field_format => 'int', :is_for_all => true)
  1712. q = IssueQuery.new
  1713. assert_include "cf_#{field.id}".to_sym, q.available_totalable_columns.map(&:name)
  1714. end
  1715. def test_available_totalable_columns_should_include_float_custom_field
  1716. field = IssueCustomField.generate!(:field_format => 'float', :is_for_all => true)
  1717. q = IssueQuery.new
  1718. assert_include "cf_#{field.id}".to_sym, q.available_totalable_columns.map(&:name)
  1719. end
  1720. def test_available_totalable_columns_should_sort_in_position_order_for_custom_field
  1721. ProjectCustomField.delete_all
  1722. cf_pos3 = ProjectCustomField.generate!(:position => 3, :is_for_all => true, :field_format => 'int')
  1723. cf_pos4 = ProjectCustomField.generate!(:position => 4, :is_for_all => true, :field_format => 'float')
  1724. cf_pos1 = ProjectCustomField.generate!(:position => 1, :is_for_all => true, :field_format => 'float')
  1725. cf_pos2 = ProjectCustomField.generate!(:position => 2, :is_for_all => true, :field_format => 'int')
  1726. q = ProjectQuery.new
  1727. custom_field_columns = q.available_totalable_columns.select{|column| column.is_a?(QueryCustomFieldColumn)}
  1728. assert_equal [cf_pos1, cf_pos2, cf_pos3, cf_pos4], custom_field_columns.collect(&:custom_field)
  1729. IssueCustomField.delete_all
  1730. cf_pos3 = IssueCustomField.generate!(:position => 3, :is_for_all => true, :field_format => 'int')
  1731. cf_pos4 = IssueCustomField.generate!(:position => 4, :is_for_all => true, :field_format => 'float')
  1732. cf_pos1 = IssueCustomField.generate!(:position => 1, :is_for_all => true, :field_format => 'float')
  1733. cf_pos2 = IssueCustomField.generate!(:position => 2, :is_for_all => true, :field_format => 'int')
  1734. q = IssueQuery.new
  1735. custom_field_columns = q.available_totalable_columns.select{|column| column.is_a?(QueryCustomFieldColumn)}
  1736. assert_equal [cf_pos1, cf_pos2, cf_pos3, cf_pos4], custom_field_columns.collect(&:custom_field)
  1737. ProjectCustomField.delete_all
  1738. IssueCustomField.delete_all
  1739. TimeEntryCustomField.delete_all
  1740. cf_pos3 = TimeEntryCustomField.generate!(:position => 3, :is_for_all => true, :field_format => 'int')
  1741. cf_pos4 = TimeEntryCustomField.generate!(:position => 4, :is_for_all => true, :field_format => 'float')
  1742. cf_pos1 = TimeEntryCustomField.generate!(:position => 1, :is_for_all => true, :field_format => 'float')
  1743. cf_pos2 = TimeEntryCustomField.generate!(:position => 2, :is_for_all => true, :field_format => 'int')
  1744. q = TimeEntryQuery.new
  1745. custom_field_columns = q.available_totalable_columns.select{|column| column.is_a?(QueryCustomFieldColumn)}
  1746. assert_equal [cf_pos1, cf_pos2, cf_pos3, cf_pos4], custom_field_columns.collect(&:custom_field)
  1747. end
  1748. def test_total_for_estimated_hours
  1749. Issue.delete_all
  1750. Issue.generate!(:estimated_hours => 5.5)
  1751. Issue.generate!(:estimated_hours => 1.1)
  1752. Issue.generate!
  1753. q = IssueQuery.new
  1754. assert_equal 6.6, q.total_for(:estimated_hours)
  1755. end
  1756. def test_total_by_group_for_estimated_hours
  1757. Issue.delete_all
  1758. Issue.generate!(:estimated_hours => 5.5, :assigned_to_id => 2)
  1759. Issue.generate!(:estimated_hours => 1.1, :assigned_to_id => 3)
  1760. Issue.generate!(:estimated_hours => 3.5)
  1761. q = IssueQuery.new(:group_by => 'assigned_to')
  1762. assert_equal(
  1763. {nil => 3.5, User.find(2) => 5.5, User.find(3) => 1.1},
  1764. q.total_by_group_for(:estimated_hours)
  1765. )
  1766. end
  1767. def test_total_for_spent_hours
  1768. TimeEntry.delete_all
  1769. TimeEntry.generate!(:hours => 5.5)
  1770. TimeEntry.generate!(:hours => 1.1)
  1771. q = IssueQuery.new
  1772. assert_equal 6.6, q.total_for(:spent_hours)
  1773. end
  1774. def test_total_by_group_for_spent_hours
  1775. TimeEntry.delete_all
  1776. TimeEntry.generate!(:hours => 5.5, :issue_id => 1)
  1777. TimeEntry.generate!(:hours => 1.1, :issue_id => 2)
  1778. Issue.where(:id => 1).update_all(:assigned_to_id => 2)
  1779. Issue.where(:id => 2).update_all(:assigned_to_id => 3)
  1780. q = IssueQuery.new(:group_by => 'assigned_to')
  1781. assert_equal(
  1782. {User.find(2) => 5.5, User.find(3) => 1.1},
  1783. q.total_by_group_for(:spent_hours)
  1784. )
  1785. end
  1786. def test_total_by_project_group_for_spent_hours
  1787. TimeEntry.delete_all
  1788. TimeEntry.generate!(:hours => 5.5, :issue_id => 1)
  1789. TimeEntry.generate!(:hours => 1.1, :issue_id => 2)
  1790. Issue.where(:id => 1).update_all(:assigned_to_id => 2)
  1791. Issue.where(:id => 2).update_all(:assigned_to_id => 3)
  1792. q = IssueQuery.new(:group_by => 'project')
  1793. assert_equal(
  1794. {Project.find(1) => 6.6},
  1795. q.total_by_group_for(:spent_hours)
  1796. )
  1797. end
  1798. def test_total_for_int_custom_field
  1799. field = IssueCustomField.generate!(:field_format => 'int', :is_for_all => true)
  1800. CustomValue.create!(:customized => Issue.find(1), :custom_field => field, :value => '2')
  1801. CustomValue.create!(:customized => Issue.find(2), :custom_field => field, :value => '7')
  1802. CustomValue.create!(:customized => Issue.find(3), :custom_field => field, :value => '')
  1803. q = IssueQuery.new
  1804. assert_equal 9, q.total_for("cf_#{field.id}")
  1805. end
  1806. def test_total_by_group_for_int_custom_field
  1807. field = IssueCustomField.generate!(:field_format => 'int', :is_for_all => true)
  1808. CustomValue.create!(:customized => Issue.find(1), :custom_field => field, :value => '2')
  1809. CustomValue.create!(:customized => Issue.find(2), :custom_field => field, :value => '7')
  1810. Issue.where(:id => 1).update_all(:assigned_to_id => 2)
  1811. Issue.where(:id => 2).update_all(:assigned_to_id => 3)
  1812. q = IssueQuery.new(:group_by => 'assigned_to')
  1813. assert_equal(
  1814. {User.find(2) => 2, User.find(3) => 7},
  1815. q.total_by_group_for("cf_#{field.id}")
  1816. )
  1817. end
  1818. def test_total_for_float_custom_field
  1819. field = IssueCustomField.generate!(:field_format => 'float', :is_for_all => true)
  1820. CustomValue.create!(:customized => Issue.find(1), :custom_field => field, :value => '2.3')
  1821. CustomValue.create!(:customized => Issue.find(2), :custom_field => field, :value => '7')
  1822. CustomValue.create!(:customized => Issue.find(3), :custom_field => field, :value => '')
  1823. q = IssueQuery.new
  1824. assert_equal 9.3, q.total_for("cf_#{field.id}")
  1825. end
  1826. def test_invalid_query_should_raise_query_statement_invalid_error
  1827. q = IssueQuery.new
  1828. assert_raise Query::StatementInvalid do
  1829. q.issues(:conditions => "foo = 1")
  1830. end
  1831. end
  1832. def test_issue_count
  1833. q = IssueQuery.new(:name => '_')
  1834. issue_count = q.issue_count
  1835. assert_equal q.issues.size, issue_count
  1836. end
  1837. def test_issue_count_with_archived_issues
  1838. p = Project.generate! do |project|
  1839. project.status = Project::STATUS_ARCHIVED
  1840. end
  1841. i = Issue.generate!(:project => p, :tracker => p.trackers.first)
  1842. assert !i.visible?
  1843. test_issue_count
  1844. end
  1845. def test_issue_count_by_association_group
  1846. q = IssueQuery.new(:name => '_', :group_by => 'assigned_to')
  1847. count_by_group = q.result_count_by_group
  1848. assert_kind_of Hash, count_by_group
  1849. assert_equal %w(NilClass User), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
  1850. assert_equal %w(Integer), count_by_group.values.collect {|k| k.class.name}.uniq
  1851. assert count_by_group.has_key?(User.find(3))
  1852. end
  1853. def test_issue_count_by_list_custom_field_group
  1854. q = IssueQuery.new(:name => '_', :group_by => 'cf_1')
  1855. count_by_group = q.result_count_by_group
  1856. assert_kind_of Hash, count_by_group
  1857. assert_equal %w(NilClass String), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
  1858. assert_equal %w(Integer), count_by_group.values.collect {|k| k.class.name}.uniq
  1859. assert count_by_group.has_key?('MySQL')
  1860. end
  1861. def test_issue_count_by_date_custom_field_group
  1862. q = IssueQuery.new(:name => '_', :group_by => 'cf_8')
  1863. count_by_group = q.result_count_by_group
  1864. assert_kind_of Hash, count_by_group
  1865. assert_equal %w(Date NilClass), count_by_group.keys.collect {|k| k.class.name}.uniq.sort
  1866. assert_equal %w(Integer), count_by_group.values.collect {|k| k.class.name}.uniq
  1867. end
  1868. def test_issue_count_with_nil_group_only
  1869. Issue.update_all("assigned_to_id = NULL")
  1870. q = IssueQuery.new(:name => '_', :group_by => 'assigned_to')
  1871. count_by_group = q.result_count_by_group
  1872. assert_kind_of Hash, count_by_group
  1873. assert_equal 1, count_by_group.keys.size
  1874. assert_nil count_by_group.keys.first
  1875. end
  1876. def test_issue_ids
  1877. q = IssueQuery.new(:name => '_')
  1878. q.sort_criteria = ['subject', 'id']
  1879. issues = q.issues
  1880. assert_equal issues.map(&:id), q.issue_ids
  1881. end
  1882. def test_label_for
  1883. set_language_if_valid 'en'
  1884. q = IssueQuery.new
  1885. assert_equal 'Assignee', q.label_for('assigned_to_id')
  1886. end
  1887. def test_label_for_fr
  1888. set_language_if_valid 'fr'
  1889. q = IssueQuery.new
  1890. assert_equal 'Assigné à', q.label_for('assigned_to_id')
  1891. end
  1892. def test_editable_by
  1893. admin = User.find(1)
  1894. manager = User.find(2)
  1895. developer = User.find(3)
  1896. # Public query on project 1
  1897. q = IssueQuery.find(1)
  1898. assert q.editable_by?(admin)
  1899. assert q.editable_by?(manager)
  1900. assert !q.editable_by?(developer)
  1901. # Private query on project 1
  1902. q = IssueQuery.find(2)
  1903. assert q.editable_by?(admin)
  1904. assert !q.editable_by?(manager)
  1905. assert q.editable_by?(developer)
  1906. # Private query for all projects
  1907. q = IssueQuery.find(3)
  1908. assert q.editable_by?(admin)
  1909. assert !q.editable_by?(manager)
  1910. assert q.editable_by?(developer)
  1911. end
  1912. def test_editable_by_for_global_query
  1913. admin = User.find(1)
  1914. manager = User.find(2)
  1915. developer = User.find(3)
  1916. q = IssueQuery.find(4)
  1917. assert q.editable_by?(admin)
  1918. assert !q.editable_by?(manager)
  1919. assert !q.editable_by?(developer)
  1920. end
  1921. def test_editable_by_for_global_query_with_project_set
  1922. admin = User.find(1)
  1923. manager = User.find(2)
  1924. developer = User.find(3)
  1925. q = IssueQuery.find(4)
  1926. q.project = Project.find(1)
  1927. assert q.editable_by?(admin)
  1928. assert !q.editable_by?(manager)
  1929. assert !q.editable_by?(developer)
  1930. end
  1931. def test_visible_scope
  1932. query_ids = IssueQuery.visible(User.anonymous).map(&:id)
  1933. assert query_ids.include?(1), 'public query on public project was not visible'
  1934. assert query_ids.include?(4), 'public query for all projects was not visible'
  1935. assert !query_ids.include?(2), 'private query on public project was visible'
  1936. assert !query_ids.include?(3), 'private query for all projects was visible'
  1937. assert !query_ids.include?(7), 'public query on private project was visible'
  1938. end
  1939. def test_query_with_public_visibility_should_be_visible_to_anyone
  1940. q = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_PUBLIC)
  1941. assert q.visible?(User.anonymous)
  1942. assert IssueQuery.visible(User.anonymous).find_by_id(q.id)
  1943. assert q.visible?(User.find(7))
  1944. assert IssueQuery.visible(User.find(7)).find_by_id(q.id)
  1945. assert q.visible?(User.find(2))
  1946. assert IssueQuery.visible(User.find(2)).find_by_id(q.id)
  1947. assert q.visible?(User.find(1))
  1948. assert IssueQuery.visible(User.find(1)).find_by_id(q.id)
  1949. end
  1950. def test_query_with_roles_visibility_should_be_visible_to_user_with_role
  1951. q = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_ROLES, :role_ids => [1, 2])
  1952. assert !q.visible?(User.anonymous)
  1953. assert_nil IssueQuery.visible(User.anonymous).find_by_id(q.id)
  1954. assert !q.visible?(User.find(7))
  1955. assert_nil IssueQuery.visible(User.find(7)).find_by_id(q.id)
  1956. assert q.visible?(User.find(2))
  1957. assert IssueQuery.visible(User.find(2)).find_by_id(q.id)
  1958. assert q.visible?(User.find(1))
  1959. assert IssueQuery.visible(User.find(1)).find_by_id(q.id)
  1960. # Should ignore archived project memberships
  1961. Project.find(1).archive
  1962. assert !q.visible?(User.find(3))
  1963. assert_nil IssueQuery.visible(User.find(3)).find_by_id(q.id)
  1964. end
  1965. def test_query_with_private_visibility_should_be_visible_to_owner
  1966. q = IssueQuery.create!(:name => 'Query', :visibility => IssueQuery::VISIBILITY_PRIVATE, :user => User.find(7))
  1967. assert !q.visible?(User.anonymous)
  1968. assert_nil IssueQuery.visible(User.anonymous).find_by_id(q.id)
  1969. assert q.visible?(User.find(7))
  1970. assert IssueQuery.visible(User.find(7)).find_by_id(q.id)
  1971. assert !q.visible?(User.find(2))
  1972. assert_nil IssueQuery.visible(User.find(2)).find_by_id(q.id)
  1973. assert q.visible?(User.find(1))
  1974. assert_nil IssueQuery.visible(User.find(1)).find_by_id(q.id)
  1975. end
  1976. def test_build_from_params_should_not_update_query_with_nil_param_values
  1977. q =
  1978. IssueQuery.create!(
  1979. :name => 'Query',
  1980. :type => "IssueQuery",
  1981. :user => User.find(7),
  1982. :filters => {"status_id" => {:values => ["1"], :operator => "o"}},
  1983. :column_names => [:tracker, :status],
  1984. :sort_criteria => ['id', 'asc'],
  1985. :group_by => "project",
  1986. :options => {
  1987. :totalable_names=>[:estimated_hours],
  1988. :draw_relations => '1',
  1989. :draw_progress_line => '1'
  1990. }
  1991. )
  1992. old_attributes = q.attributes
  1993. q.build_from_params({})
  1994. assert_equal old_attributes, q.attributes
  1995. end
  1996. test "#available_filters should include users of visible projects in cross-project view" do
  1997. users = IssueQuery.new.available_filters["assigned_to_id"]
  1998. assert_not_nil users
  1999. assert users[:values].pluck(1).include?("3")
  2000. end
  2001. test "#available_filters should include users of subprojects" do
  2002. user1 = User.generate!
  2003. user2 = User.generate!
  2004. project = Project.find(1)
  2005. Member.create!(:principal => user1, :project => project.children.visible.first, :role_ids => [1])
  2006. users = IssueQuery.new(:project => project).available_filters["assigned_to_id"]
  2007. assert_not_nil users
  2008. assert users[:values].pluck(1).include?(user1.id.to_s)
  2009. assert !users[:values].pluck(1).include?(user2.id.to_s)
  2010. end
  2011. test "#available_filters should include visible projects in cross-project view" do
  2012. projects = IssueQuery.new.available_filters["project_id"]
  2013. assert_not_nil projects
  2014. assert projects[:values].pluck(1).include?("1")
  2015. end
  2016. test "#available_filters should include 'member_of_group' filter" do
  2017. query = IssueQuery.new
  2018. assert query.available_filters.key?("member_of_group")
  2019. assert_equal :list_optional, query.available_filters["member_of_group"][:type]
  2020. assert query.available_filters["member_of_group"][:values].present?
  2021. assert_equal Group.givable.sort.map {|g| [g.name, g.id.to_s]},
  2022. query.available_filters["member_of_group"][:values].sort
  2023. end
  2024. test "#available_filters should include 'assigned_to_role' filter" do
  2025. query = IssueQuery.new
  2026. assert query.available_filters.key?("assigned_to_role")
  2027. assert_equal :list_optional, query.available_filters["assigned_to_role"][:type]
  2028. assert query.available_filters["assigned_to_role"][:values].include?(['Manager', '1'])
  2029. assert query.available_filters["assigned_to_role"][:values].include?(['Developer', '2'])
  2030. assert query.available_filters["assigned_to_role"][:values].include?(['Reporter', '3'])
  2031. assert ! query.available_filters["assigned_to_role"][:values].include?(['Non member', '4'])
  2032. assert ! query.available_filters["assigned_to_role"][:values].include?(['Anonymous', '5'])
  2033. end
  2034. def test_available_filters_should_include_custom_field_according_to_user_visibility
  2035. visible_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => true)
  2036. hidden_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => false, :role_ids => [1])
  2037. with_current_user User.find(3) do
  2038. query = IssueQuery.new
  2039. assert_include "cf_#{visible_field.id}", query.available_filters.keys
  2040. assert_not_include "cf_#{hidden_field.id}", query.available_filters.keys
  2041. end
  2042. end
  2043. def test_available_columns_should_include_custom_field_according_to_user_visibility
  2044. visible_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => true)
  2045. hidden_field = IssueCustomField.generate!(:is_for_all => true, :is_filter => true, :visible => false, :role_ids => [1])
  2046. with_current_user User.find(3) do
  2047. query = IssueQuery.new
  2048. assert_include :"cf_#{visible_field.id}", query.available_columns.map(&:name)
  2049. assert_not_include :"cf_#{hidden_field.id}", query.available_columns.map(&:name)
  2050. end
  2051. end
  2052. def test_available_columns_should_not_include_total_estimated_hours_when_trackers_disabled_estimated_hours
  2053. Tracker.visible.each do |tracker|
  2054. tracker.core_fields = tracker.core_fields.reject{|field| field == 'estimated_hours'}
  2055. tracker.save!
  2056. end
  2057. query = IssueQuery.new
  2058. available_columns = query.available_columns.map(&:name)
  2059. assert_not_include :estimated_hours, available_columns
  2060. assert_not_include :total_estimated_hours, available_columns
  2061. tracker = Tracker.visible.first
  2062. tracker.core_fields = ['estimated_hours']
  2063. tracker.save!
  2064. query = IssueQuery.new
  2065. available_columns = query.available_columns.map(&:name)
  2066. assert_include :estimated_hours, available_columns
  2067. assert_include :total_estimated_hours, available_columns
  2068. end
  2069. def setup_member_of_group
  2070. Group.destroy_all # No fixtures
  2071. @user_in_group = User.generate!
  2072. @second_user_in_group = User.generate!
  2073. @user_in_group2 = User.generate!
  2074. @user_not_in_group = User.generate!
  2075. @group = Group.generate!.reload
  2076. @group.users << @user_in_group
  2077. @group.users << @second_user_in_group
  2078. @group2 = Group.generate!.reload
  2079. @group2.users << @user_in_group2
  2080. @query = IssueQuery.new(:name => '_')
  2081. end
  2082. test "member_of_group filter should search assigned to for users in the group" do
  2083. setup_member_of_group
  2084. @query.add_filter('member_of_group', '=', [@group.id.to_s])
  2085. assert_find_issues_with_query_is_successful @query
  2086. end
  2087. test "member_of_group filter should search not assigned to any group member (none)" do
  2088. setup_member_of_group
  2089. @query.add_filter('member_of_group', '!*', [''])
  2090. assert_find_issues_with_query_is_successful @query
  2091. end
  2092. test "member_of_group filter should search assigned to any group member (all)" do
  2093. setup_member_of_group
  2094. @query.add_filter('member_of_group', '*', [''])
  2095. assert_find_issues_with_query_is_successful @query
  2096. end
  2097. test "member_of_group filter should return an empty set with = empty group" do
  2098. setup_member_of_group
  2099. @empty_group = Group.generate!
  2100. @query.add_filter('member_of_group', '=', [@empty_group.id.to_s])
  2101. assert_equal [], find_issues_with_query(@query)
  2102. end
  2103. test "member_of_group filter should return issues with ! empty group" do
  2104. setup_member_of_group
  2105. @empty_group = Group.generate!
  2106. @query.add_filter('member_of_group', '!', [@empty_group.id.to_s])
  2107. assert_find_issues_with_query_is_successful @query
  2108. end
  2109. def setup_assigned_to_role
  2110. @manager_role = Role.find_by_name('Manager')
  2111. @developer_role = Role.find_by_name('Developer')
  2112. @project = Project.generate!
  2113. @manager = User.generate!
  2114. @developer = User.generate!
  2115. @boss = User.generate!
  2116. @guest = User.generate!
  2117. User.add_to_project(@manager, @project, @manager_role)
  2118. User.add_to_project(@developer, @project, @developer_role)
  2119. User.add_to_project(@boss, @project, [@manager_role, @developer_role])
  2120. @issue1 = Issue.generate!(:project => @project, :assigned_to_id => @manager.id)
  2121. @issue2 = Issue.generate!(:project => @project, :assigned_to_id => @developer.id)
  2122. @issue3 = Issue.generate!(:project => @project, :assigned_to_id => @boss.id)
  2123. @issue4 = Issue.generate!(:project => @project, :author_id => @guest.id, :assigned_to_id => @guest.id)
  2124. @issue5 = Issue.generate!(:project => @project)
  2125. @query = IssueQuery.new(:name => '_', :project => @project)
  2126. end
  2127. test "assigned_to_role filter should search assigned to for users with the Role" do
  2128. setup_assigned_to_role
  2129. @query.add_filter('assigned_to_role', '=', [@manager_role.id.to_s])
  2130. assert_query_result [@issue1, @issue3], @query
  2131. end
  2132. test "assigned_to_role filter should search assigned to for users with the Role on the issue project" do
  2133. setup_assigned_to_role
  2134. other_project = Project.generate!
  2135. User.add_to_project(@developer, other_project, @manager_role)
  2136. @query.add_filter('assigned_to_role', '=', [@manager_role.id.to_s])
  2137. assert_query_result [@issue1, @issue3], @query
  2138. end
  2139. test "assigned_to_role filter should return an empty set with empty role" do
  2140. setup_assigned_to_role
  2141. @empty_role = Role.generate!
  2142. @query.add_filter('assigned_to_role', '=', [@empty_role.id.to_s])
  2143. assert_query_result [], @query
  2144. end
  2145. test "assigned_to_role filter should search assigned to for users without the Role" do
  2146. setup_assigned_to_role
  2147. @query.add_filter('assigned_to_role', '!', [@manager_role.id.to_s])
  2148. assert_query_result [@issue2, @issue4, @issue5], @query
  2149. end
  2150. test "assigned_to_role filter should search assigned to for users not assigned to any Role (none)" do
  2151. setup_assigned_to_role
  2152. @query.add_filter('assigned_to_role', '!*', [''])
  2153. assert_query_result [@issue4, @issue5], @query
  2154. end
  2155. test "assigned_to_role filter should search assigned to for users assigned to any Role (all)" do
  2156. setup_assigned_to_role
  2157. @query.add_filter('assigned_to_role', '*', [''])
  2158. assert_query_result [@issue1, @issue2, @issue3], @query
  2159. end
  2160. test "assigned_to_role filter should return issues with ! empty role" do
  2161. setup_assigned_to_role
  2162. @empty_role = Role.generate!
  2163. @query.add_filter('assigned_to_role', '!', [@empty_role.id.to_s])
  2164. assert_query_result [@issue1, @issue2, @issue3, @issue4, @issue5], @query
  2165. end
  2166. def test_query_column_should_accept_a_symbol_as_caption
  2167. set_language_if_valid 'en'
  2168. c = QueryColumn.new('foo', :caption => :general_text_Yes)
  2169. assert_equal 'Yes', c.caption
  2170. end
  2171. def test_query_column_should_accept_a_proc_as_caption
  2172. c = QueryColumn.new('foo', :caption => lambda {'Foo'})
  2173. assert_equal 'Foo', c.caption
  2174. end
  2175. def test_date_clause_should_respect_user_time_zone_with_local_default
  2176. @query = IssueQuery.new(:name => '_')
  2177. # user is in Hawaii (-10)
  2178. User.current = users(:users_001)
  2179. User.current.pref.update_attribute :time_zone, 'Hawaii'
  2180. # assume timestamps are stored in server local time
  2181. local_zone = Time.zone
  2182. from = Date.parse '2016-03-20'
  2183. to = Date.parse '2016-03-22'
  2184. assert c = @query.send(:date_clause, 'table', 'field', from, to, false)
  2185. # the dates should have been interpreted in the user's time zone and
  2186. # converted to local time
  2187. # what we get exactly in the sql depends on the local time zone, therefore
  2188. # it's computed here.
  2189. f = User.current.time_zone.local(from.year, from.month, from.day).yesterday.end_of_day.in_time_zone(local_zone)
  2190. t = User.current.time_zone.local(to.year, to.month, to.day).end_of_day.in_time_zone(local_zone)
  2191. assert_equal "table.field > '#{Query.connection.quoted_date f}' AND table.field <= '#{Query.connection.quoted_date t}'", c
  2192. end
  2193. def test_date_clause_should_respect_user_time_zone_with_utc_default
  2194. @query = IssueQuery.new(:name => '_')
  2195. # user is in Hawaii (-10)
  2196. User.current = users(:users_001)
  2197. User.current.pref.update_attribute :time_zone, 'Hawaii'
  2198. # assume timestamps are stored as utc
  2199. ActiveRecord::Base.default_timezone = :utc
  2200. from = Date.parse '2016-03-20'
  2201. to = Date.parse '2016-03-22'
  2202. assert c = @query.send(:date_clause, 'table', 'field', from, to, false)
  2203. # the dates should have been interpreted in the user's time zone and
  2204. # converted to utc. March 20 in Hawaii begins at 10am UTC.
  2205. f = Time.new(2016, 3, 20, 9, 59, 59, 0).end_of_hour
  2206. t = Time.new(2016, 3, 23, 9, 59, 59, 0).end_of_hour
  2207. assert_equal "table.field > '#{Query.connection.quoted_date f}' AND table.field <= '#{Query.connection.quoted_date t}'", c
  2208. ensure
  2209. ActiveRecord::Base.default_timezone = :local # restore Redmine default
  2210. end
  2211. def test_project_statement_with_closed_subprojects
  2212. project = Project.find(1)
  2213. project.descendants.each(&:close)
  2214. with_settings :display_subprojects_issues => '1' do
  2215. query = IssueQuery.new(:name => '_', :project => project)
  2216. statement = query.project_statement
  2217. assert_equal "projects.lft >= #{project.lft} AND projects.rgt <= #{project.rgt}", statement
  2218. end
  2219. end
  2220. def test_filter_on_subprojects
  2221. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  2222. filter_name = "subproject_id"
  2223. assert_include filter_name, query.available_filters.keys
  2224. # "is" operator should include issues of parent project + issues of the selected subproject
  2225. query.filters = {filter_name => {:operator => '=', :values => ['3']}}
  2226. issues = find_issues_with_query(query)
  2227. assert_equal [1, 2, 3, 5, 7, 8, 11, 12, 13, 14], issues.map(&:id).sort
  2228. # "is not" operator should include issues of parent project + issues of all active subprojects - issues of the selected subprojects
  2229. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  2230. query.filters = {filter_name => {:operator => '!', :values => ['3']}}
  2231. issues = find_issues_with_query(query)
  2232. assert_equal [1, 2, 3, 6, 7, 8, 9, 10, 11, 12], issues.map(&:id).sort
  2233. end
  2234. def test_filter_updated_on_none_should_return_issues_with_updated_on_equal_with_created_on
  2235. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  2236. query.filters = {'updated_on' => {:operator => '!*', :values => ['']}}
  2237. issues = find_issues_with_query(query)
  2238. assert_equal [3, 6, 7, 8, 9, 10, 14], issues.map(&:id).sort
  2239. end
  2240. def test_filter_updated_on_any_should_return_issues_with_updated_on_greater_than_created_on
  2241. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  2242. query.filters = {'updated_on' => {:operator => '*', :values => ['']}}
  2243. issues = find_issues_with_query(query)
  2244. assert_equal [1, 2, 5, 11, 12, 13], issues.map(&:id).sort
  2245. end
  2246. def test_issue_statuses_should_return_only_statuses_used_by_that_project
  2247. query = IssueQuery.new(:name => '_', :project => Project.find(1))
  2248. query.filters = {'status_id' => {:operator => '=', :values => []}}
  2249. WorkflowTransition.delete_all
  2250. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 1, :new_status_id => 3)
  2251. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 1, :new_status_id => 4)
  2252. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 2, :new_status_id => 3)
  2253. WorkflowTransition.create(:role_id => 1, :tracker_id => 2, :old_status_id => 1, :new_status_id => 3)
  2254. assert_equal ['1', '2', '3', '4'], query.available_filters['status_id'][:values].map(&:second)
  2255. end
  2256. def test_issue_statuses_without_project_should_return_all_statuses
  2257. query = IssueQuery.new(:name => '_')
  2258. query.filters = {'status_id' => {:operator => '=', :values => []}}
  2259. WorkflowTransition.delete_all
  2260. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 1, :new_status_id => 3)
  2261. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 1, :new_status_id => 4)
  2262. WorkflowTransition.create(:role_id => 1, :tracker_id => 1, :old_status_id => 2, :new_status_id => 3)
  2263. WorkflowTransition.create(:role_id => 1, :tracker_id => 2, :old_status_id => 1, :new_status_id => 3)
  2264. assert_equal ['1', '2', '3', '4', '5', '6'], query.available_filters['status_id'][:values].map(&:second)
  2265. end
  2266. def test_project_status_filter_should_be_available_in_global_queries
  2267. query = IssueQuery.new(:project => nil, :name => '_')
  2268. assert query.available_filters.has_key?('project.status')
  2269. end
  2270. def test_project_status_filter_should_be_available_when_project_has_subprojects
  2271. query = IssueQuery.new(:project => Project.find(1), :name => '_')
  2272. assert query.available_filters.has_key?('project.status')
  2273. end
  2274. def test_project_status_filter_should_not_be_available_when_project_is_leaf
  2275. query = IssueQuery.new(:project => Project.find(2), :name => '_')
  2276. assert !query.available_filters.has_key?('project.status')
  2277. end
  2278. def test_project_statuses_values_should_return_only_active_and_closed_statuses
  2279. set_language_if_valid 'en'
  2280. query = IssueQuery.new(:project => nil, :name => '_')
  2281. project_status_filter = query.available_filters['project.status']
  2282. assert_not_nil project_status_filter
  2283. assert_equal [["active", "1"], ["closed", "5"]], project_status_filter[:values]
  2284. end
  2285. def test_as_params_should_serialize_query
  2286. query = IssueQuery.new(name: "_")
  2287. query.add_filter('subject', '!~', ['asdf'])
  2288. query.group_by = 'tracker'
  2289. query.totalable_names = %w(estimated_hours)
  2290. query.column_names = %w(id subject estimated_hours)
  2291. assert hsh = query.as_params
  2292. new_query = IssueQuery.build_from_params(hsh)
  2293. assert_equal query.filters, new_query.filters
  2294. assert_equal query.group_by, new_query.group_by
  2295. assert_equal query.column_names, new_query.column_names
  2296. assert_equal query.totalable_names, new_query.totalable_names
  2297. end
  2298. def test_issue_query_filter_by_spent_time
  2299. query = IssueQuery.new(:name => '_')
  2300. query.filters = {'spent_time' => {:operator => '*', :values => ['']}}
  2301. assert_equal [3, 1], query.issues.pluck(:id)
  2302. query.filters = {'spent_time' => {:operator => '!*', :values => ['']}}
  2303. assert_equal [13, 12, 11, 8, 7, 5, 2], query.issues.pluck(:id)
  2304. query.filters = {'spent_time' => {:operator => '>=', :values => ['10']}}
  2305. assert_equal [1], query.issues.pluck(:id)
  2306. query.filters = {'spent_time' => {:operator => '<=', :values => ['10']}}
  2307. assert_equal [13, 12, 11, 8, 7, 5, 3, 2], query.issues.pluck(:id)
  2308. query.filters = {'spent_time' => {:operator => '><', :values => ['1', '2']}}
  2309. assert_equal [3], query.issues.pluck(:id)
  2310. end
  2311. def test_issues_should_be_in_the_same_order_when_paginating
  2312. q = IssueQuery.new
  2313. q.sort_criteria = {'0' => ['priority', 'desc']}
  2314. issue_ids = q.issues.pluck(:id)
  2315. paginated_issue_ids = []
  2316. # Test with a maximum of 2 records per page.
  2317. ((q.issue_count / 2) + 1).times do |i|
  2318. paginated_issue_ids += q.issues(:offset => (i * 2), :limit => 2).pluck(:id)
  2319. end
  2320. # Non-paginated issue ids and paginated issue ids should be in the same order.
  2321. assert_equal issue_ids, paginated_issue_ids
  2322. end
  2323. def test_destruction_of_default_query_should_remove_reference_from_project
  2324. project = Project.find('ecookbook')
  2325. project_query = IssueQuery.find(1)
  2326. project.update_column :default_issue_query_id, project_query.id
  2327. project_query.destroy
  2328. project.reload
  2329. assert_nil project.default_issue_query_id
  2330. end
  2331. def test_should_determine_default_issue_query
  2332. project = Project.find('ecookbook')
  2333. user = project.users.first
  2334. project_query = IssueQuery.find(1)
  2335. query = IssueQuery.find(4)
  2336. user_query = IssueQuery.find(3)
  2337. user_query.update(visibility: Query::VISIBILITY_PUBLIC)
  2338. user_query.update_column :user_id, user.id
  2339. [nil, user, User.anonymous].each do |u|
  2340. [nil, project].each do |p|
  2341. assert_nil IssueQuery.default(project: p, user: u)
  2342. end
  2343. end
  2344. # only global default is set
  2345. with_settings :default_issue_query => query.id do
  2346. [nil, user, User.anonymous].each do |u|
  2347. [nil, project].each do |p|
  2348. assert_equal query, IssueQuery.default(project: p, user: u)
  2349. end
  2350. end
  2351. end
  2352. # with project default
  2353. assert_equal project.id, project_query.project_id
  2354. project.update_column :default_issue_query_id, project_query.id
  2355. [nil, user, User.anonymous].each do |u|
  2356. assert_nil IssueQuery.default(project: nil, user: u)
  2357. assert_equal project_query, IssueQuery.default(project: project, user: u)
  2358. end
  2359. # project default should override global default
  2360. with_settings :default_issue_query => query.id do
  2361. [nil, user, User.anonymous].each do |u|
  2362. assert_equal query, IssueQuery.default(project: nil, user: u)
  2363. assert_equal project_query, IssueQuery.default(project: project, user: u)
  2364. end
  2365. end
  2366. # user default, overrides project and global default
  2367. user.pref.default_issue_query = user_query.id
  2368. user.pref.save
  2369. with_settings :default_issue_query => query.id do
  2370. [nil, project].each do |p|
  2371. assert_equal user_query, IssueQuery.default(project: p, user: user)
  2372. assert_equal user_query, IssueQuery.default(project: p, user: user)
  2373. end
  2374. end
  2375. end
  2376. def test_sql_contains_should_escape_value
  2377. i = Issue.generate! subject: 'Sanitize test'
  2378. query = IssueQuery.new(:project => nil, :name => '_')
  2379. query.add_filter('subject', '~', ['te%t'])
  2380. assert_equal 0, query.issue_count
  2381. i.update_column :subject, 'Sanitize te%t'
  2382. assert_equal 1, query.issue_count
  2383. i.update_column :subject, 'Sanitize te_t'
  2384. query = IssueQuery.new(:project => nil, :name => '_')
  2385. query.add_filter('subject', '~', ['te_t'])
  2386. assert_equal 1, query.issue_count
  2387. end
  2388. def test_sql_contains_should_tokenize
  2389. query = IssueQuery.new(:project => nil, :name => '_')
  2390. query.add_filter('subject', '~', ['issue today'])
  2391. assert_equal 1, query.issue_count
  2392. end
  2393. def test_display_type_should_accept_known_types
  2394. query = ProjectQuery.new(:name => '_')
  2395. query.display_type = 'list'
  2396. assert_equal 'list', query.display_type
  2397. end
  2398. def test_display_type_should_not_accept_unknown_types
  2399. query = ProjectQuery.new(:name => '_')
  2400. query.display_type = 'invalid'
  2401. assert_equal 'board', query.display_type
  2402. end
  2403. end