##// END OF EJS Templates
Make sure we don't cast an empty string to numeric (#12713)....
Jean-Philippe Lang -
r10873:77f6b404fac5
parent child
Show More
@@ -185,7 +185,7 class CustomField < ActiveRecord::Base
185 # Make the database cast values into numeric
185 # Make the database cast values into numeric
186 # Postgresql will raise an error if a value can not be casted!
186 # Postgresql will raise an error if a value can not be casted!
187 # CustomValue validations should ensure that it doesn't occur
187 # CustomValue validations should ensure that it doesn't occur
188 "CAST(#{join_alias}.value AS decimal(30,3))"
188 "CAST(CASE #{join_alias}.value WHEN '' THEN '0' ELSE #{join_alias}.value END AS decimal(30,3))"
189 when 'user', 'version'
189 when 'user', 'version'
190 value_class.fields_for_order_statement(value_join_alias)
190 value_class.fields_for_order_statement(value_join_alias)
191 else
191 else
@@ -220,7 +220,7 class CustomField < ActiveRecord::Base
220 " AND #{join_alias}_2.customized_id = #{join_alias}.customized_id" +
220 " AND #{join_alias}_2.customized_id = #{join_alias}.customized_id" +
221 " AND #{join_alias}_2.custom_field_id = #{join_alias}.custom_field_id)" +
221 " AND #{join_alias}_2.custom_field_id = #{join_alias}.custom_field_id)" +
222 " LEFT OUTER JOIN #{value_class.table_name} #{value_join_alias}" +
222 " LEFT OUTER JOIN #{value_class.table_name} #{value_join_alias}" +
223 " ON CAST(#{join_alias}.value as decimal(30,0)) = #{value_join_alias}.id"
223 " ON CAST(CASE #{join_alias}.value WHEN '' THEN '0' ELSE #{join_alias}.value END AS decimal(30,0)) = #{value_join_alias}.id"
224 when 'int', 'float'
224 when 'int', 'float'
225 "LEFT OUTER JOIN #{CustomValue.table_name} #{join_alias}" +
225 "LEFT OUTER JOIN #{CustomValue.table_name} #{join_alias}" +
226 " ON #{join_alias}.customized_type = '#{self.class.customized_class.base_class.name}'" +
226 " ON #{join_alias}.customized_type = '#{self.class.customized_class.base_class.name}'" +
@@ -532,13 +532,13 class Query < ActiveRecord::Base
532 sql = date_clause(db_table, db_field, (Date.parse(value.first) rescue nil), (Date.parse(value.first) rescue nil))
532 sql = date_clause(db_table, db_field, (Date.parse(value.first) rescue nil), (Date.parse(value.first) rescue nil))
533 when :integer
533 when :integer
534 if is_custom_filter
534 if is_custom_filter
535 sql = "(#{db_table}.#{db_field} <> '' AND CAST(#{db_table}.#{db_field} AS decimal(30,3)) = #{value.first.to_i})"
535 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) = #{value.first.to_i})"
536 else
536 else
537 sql = "#{db_table}.#{db_field} = #{value.first.to_i}"
537 sql = "#{db_table}.#{db_field} = #{value.first.to_i}"
538 end
538 end
539 when :float
539 when :float
540 if is_custom_filter
540 if is_custom_filter
541 sql = "(#{db_table}.#{db_field} <> '' AND CAST(#{db_table}.#{db_field} AS decimal(30,3)) BETWEEN #{value.first.to_f - 1e-5} AND #{value.first.to_f + 1e-5})"
541 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) BETWEEN #{value.first.to_f - 1e-5} AND #{value.first.to_f + 1e-5})"
542 else
542 else
543 sql = "#{db_table}.#{db_field} BETWEEN #{value.first.to_f - 1e-5} AND #{value.first.to_f + 1e-5}"
543 sql = "#{db_table}.#{db_field} BETWEEN #{value.first.to_f - 1e-5} AND #{value.first.to_f + 1e-5}"
544 end
544 end
@@ -567,7 +567,7 class Query < ActiveRecord::Base
567 sql = date_clause(db_table, db_field, (Date.parse(value.first) rescue nil), nil)
567 sql = date_clause(db_table, db_field, (Date.parse(value.first) rescue nil), nil)
568 else
568 else
569 if is_custom_filter
569 if is_custom_filter
570 sql = "(#{db_table}.#{db_field} <> '' AND CAST(#{db_table}.#{db_field} AS decimal(30,3)) >= #{value.first.to_f})"
570 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) >= #{value.first.to_f})"
571 else
571 else
572 sql = "#{db_table}.#{db_field} >= #{value.first.to_f}"
572 sql = "#{db_table}.#{db_field} >= #{value.first.to_f}"
573 end
573 end
@@ -577,7 +577,7 class Query < ActiveRecord::Base
577 sql = date_clause(db_table, db_field, nil, (Date.parse(value.first) rescue nil))
577 sql = date_clause(db_table, db_field, nil, (Date.parse(value.first) rescue nil))
578 else
578 else
579 if is_custom_filter
579 if is_custom_filter
580 sql = "(#{db_table}.#{db_field} <> '' AND CAST(#{db_table}.#{db_field} AS decimal(30,3)) <= #{value.first.to_f})"
580 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) <= #{value.first.to_f})"
581 else
581 else
582 sql = "#{db_table}.#{db_field} <= #{value.first.to_f}"
582 sql = "#{db_table}.#{db_field} <= #{value.first.to_f}"
583 end
583 end
@@ -587,7 +587,7 class Query < ActiveRecord::Base
587 sql = date_clause(db_table, db_field, (Date.parse(value[0]) rescue nil), (Date.parse(value[1]) rescue nil))
587 sql = date_clause(db_table, db_field, (Date.parse(value[0]) rescue nil), (Date.parse(value[1]) rescue nil))
588 else
588 else
589 if is_custom_filter
589 if is_custom_filter
590 sql = "(#{db_table}.#{db_field} <> '' AND CAST(#{db_table}.#{db_field} AS decimal(30,3)) BETWEEN #{value[0].to_f} AND #{value[1].to_f})"
590 sql = "(#{db_table}.#{db_field} <> '' AND CAST(CASE #{db_table}.#{db_field} WHEN '' THEN '0' ELSE #{db_table}.#{db_field} END AS decimal(30,3)) BETWEEN #{value[0].to_f} AND #{value[1].to_f})"
591 else
591 else
592 sql = "#{db_table}.#{db_field} BETWEEN #{value[0].to_f} AND #{value[1].to_f}"
592 sql = "#{db_table}.#{db_field} BETWEEN #{value[0].to_f} AND #{value[1].to_f}"
593 end
593 end
@@ -328,7 +328,7 class QueryTest < ActiveSupport::TestCase
328 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
328 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
329 query = IssueQuery.new(:project => Project.find(1), :name => '_')
329 query = IssueQuery.new(:project => Project.find(1), :name => '_')
330 query.add_filter("cf_#{f.id}", '<=', ['30'])
330 query.add_filter("cf_#{f.id}", '<=', ['30'])
331 assert query.statement.include?("CAST(custom_values.value AS decimal(30,3)) <= 30.0")
331 assert_match /CAST.+ <= 30\.0/, query.statement
332 find_issues_with_query(query)
332 find_issues_with_query(query)
333 end
333 end
334
334
@@ -343,7 +343,7 class QueryTest < ActiveSupport::TestCase
343 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
343 f = IssueCustomField.create!(:name => 'filter', :field_format => 'int', :is_filter => true, :is_for_all => true)
344 query = IssueQuery.new(:project => Project.find(1), :name => '_')
344 query = IssueQuery.new(:project => Project.find(1), :name => '_')
345 query.add_filter("cf_#{f.id}", '><', ['30', '40'])
345 query.add_filter("cf_#{f.id}", '><', ['30', '40'])
346 assert_include "CAST(custom_values.value AS decimal(30,3)) BETWEEN 30.0 AND 40.0", query.statement
346 assert_match /CAST.+ BETWEEN 30.0 AND 40.0/, query.statement
347 find_issues_with_query(query)
347 find_issues_with_query(query)
348 end
348 end
349
349
General Comments 0
You need to be logged in to leave comments. Login now