Class: RuboCop::Cop::Lint::LiteralInInterpolation
| Relationships & Source Files | |
| Super Chains via Extension / Inclusion / Inheritance | |
| 
         Class Chain: 
        
          self,
           
      ::RuboCop::Cop::AutoCorrector,
          ::RuboCop::Cop::Base,
          ::RuboCop::ExcludeLimit,
          NodePattern::Macros,
          RuboCop::AST::Sexp
         | 
    |
| 
         Instance Chain: 
        
       | 
    |
| Inherits: | 
        RuboCop::Cop::Base
        
  | 
    
| Defined in: | lib/rubocop/cop/lint/literal_in_interpolation.rb | 
Overview
Checks for interpolated literals.
| 
 Note 
 | 
Array literals interpolated in regexps are not handled by this cop, but
by Lint/ArrayLiteralInRegexp instead.
 | 
Constant Summary
- 
    COMPOSITE =
    
# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 25%i[array hash pair irange erange].freeze
 - 
    MSG =
    
# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 24'Literal interpolation detected.' 
::RuboCop::Cop::Base - Inherited
  EMPTY_OFFENSES, RESTRICT_ON_SEND
::RuboCop::Cop::RangeHelp - Included
  
Class Attribute Summary
::RuboCop::Cop::AutoCorrector - Extended
::RuboCop::Cop::Base - Inherited
| .gem_requirements, .lint?, | |
| .support_autocorrect? | Returns if class supports autocorrect.  | 
    
| .support_multiple_source? | Override if your cop should be called repeatedly for multiple investigations Between calls to   | 
    
Class Method Summary
::RuboCop::Cop::Base - Inherited
| .autocorrect_incompatible_with | List of cops that should not try to autocorrect at the same time as this cop.  | 
    
| .badge | Naming.  | 
    
| .callbacks_needed, .cop_name, .department, | |
| .documentation_url | Returns a url to view this cops documentation online.  | 
    
| .exclude_from_registry | Call for abstract Cop classes.  | 
    
| .inherited, | |
| .joining_forces | Override and return the Force class(es) you need to join.  | 
    
| .match? | Returns true if the cop name or the cop namespace matches any of the given names.  | 
    
| .new, | |
| .requires_gem | Register a version requirement for the given gem name.  | 
    
| .restrict_on_send | |
::RuboCop::ExcludeLimit - Extended
| exclude_limit | Sets up a configuration option to have an exclude limit tracked.  | 
    
| transform | |
Instance Attribute Summary
::RuboCop::Cop::Base - Inherited
::RuboCop::Cop::AutocorrectLogic - Included
Instance Method Summary
- #on_interpolation(begin_node)
 - #array_in_regexp?(node) ⇒ Boolean private
 - 
    
      #autocorrected_value(node)  
    
    private
    
Metrics/CyclomaticComplexity.
 - #autocorrected_value_for_array(node) private
 - #autocorrected_value_for_hash(node) private
 - #autocorrected_value_for_string(node) private
 - #autocorrected_value_for_symbol(node) private
 - 
    
      #autocorrected_value_in_hash(node)  
    
    private
    
Metrics/AbcSize.
 - #autocorrected_value_in_hash_for_symbol(node) private
 - #ends_heredoc_line?(node) ⇒ Boolean private
 - 
    
      #handle_special_regexp_chars(begin_node, value)  
    
    private
    
Metrics/CyclomaticComplexity.
 - #in_array_percent_literal?(node) ⇒ Boolean private
 - #offending?(node) ⇒ Boolean private
 - 
    
      #prints_as_self?(node)  ⇒ Boolean 
    
    private
    
Does node print its own source when converted to a string?
 - #space_literal?(node) ⇒ Boolean private
 - #special_keyword?(node) ⇒ Boolean private
 
::RuboCop::Cop::PercentLiteral - Included
::RuboCop::Cop::RangeHelp - Included
| #add_range, | |
| #arguments_range | A range containing the first to the last argument of a method call or method definition.  | 
    
| #column_offset_between, | |
| #contents_range | A range containing only the contents of a literal with delimiters (e.g.  | 
    
| #directions, | |
| #effective_column | Returns the column attribute of the range, except if the range is on the first line and there’s a byte order mark at the beginning of that line, in which case 1 is subtracted from the column value.  | 
    
| #final_pos, #move_pos, #move_pos_str, #range_between, #range_by_whole_lines, #range_with_comments, #range_with_comments_and_lines, #range_with_surrounding_comma, #range_with_surrounding_space, #source_range | |
::RuboCop::Cop::Interpolation - Included
| #on_dstr, | |
| #on_dsym | Alias for Interpolation#on_dstr.  | 
    
| #on_interpolation | Inspect the   | 
    
| #on_node_with_interpolations, | |
| #on_regexp | Alias for Interpolation#on_dstr.  | 
    
| #on_xstr | Alias for Interpolation#on_dstr.  | 
    
::RuboCop::Cop::Base - Inherited
| #add_global_offense | Adds an offense that has no particular location.  | 
    
| #add_offense | Adds an offense on the specified range (or node with an expression) Unless that offense is disabled for this range, a corrector will be yielded to provide the cop the opportunity to autocorrect the offense.  | 
    
| #begin_investigation | Called before any investigation.  | 
    
| #callbacks_needed, | |
| #cop_config | Configuration Helpers.  | 
    
| #cop_name, #excluded_file?, | |
| #external_dependency_checksum | This method should be overridden when a cop’s behavior depends on state that lives outside of these locations:  | 
    
| #inspect, | |
| #message | Gets called if no message is specified when calling   | 
    
| #name | Alias for Base#cop_name.  | 
    
| #offenses, | |
| #on_investigation_end | Called after all on_…  | 
    
| #on_new_investigation | Called before all on_…  | 
    
| #on_other_file | Called instead of all on_…  | 
    
| #parse | There should be very limited reasons for a Cop to do it’s own parsing.  | 
    
| #parser_engine, | |
| #ready | Called between investigations.  | 
    
| #relevant_file?, | |
| #target_gem_version | Returns a gems locked versions (i.e.  | 
    
| #target_rails_version, #target_ruby_version, #annotate, #apply_correction, #attempt_correction, | |
| #callback_argument | Reserved for Cop::Cop.  | 
    
| #complete_investigation | Called to complete an investigation.  | 
    
| #correct, #current_corrector, | |
| #current_offense_locations | Reserved for Commissioner:  | 
    
| #current_offenses, #currently_disabled_lines, #custom_severity, #default_severity, #disable_uncorrectable, #enabled_line?, #file_name_matches_any?, #find_message, #find_severity, #range_for_original, #range_from_node_or_range, | |
| #reset_investigation | Actually private methods.  | 
    
| #use_corrector | |
::RuboCop::Cop::AutocorrectLogic - Included
::RuboCop::Cop::IgnoredNode - Included
Constructor Details
This class inherits a constructor from RuboCop::Cop::Base
Instance Method Details
    #array_in_regexp?(node)  ⇒ Boolean  (private)
  
# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 72
def array_in_regexp?(node) grandparent = node.parent.parent node.array_type? && grandparent.regexp_type? end
#autocorrected_value(node) (private)
Metrics/CyclomaticComplexity
# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 78
def autocorrected_value(node) case node.type when :int node.children.last.to_i.to_s when :float node.children.last.to_f.to_s when :str autocorrected_value_for_string(node) when :sym autocorrected_value_for_symbol(node) when :array autocorrected_value_for_array(node) when :hash autocorrected_value_for_hash(node) when :nil '' else node.source.gsub('"', '\"') end end
#autocorrected_value_for_array(node) (private)
[ GitHub ]# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 145
def autocorrected_value_for_array(node) return node.source.gsub('"', '\"') unless node.percent_literal? contents_range(node).source.split.to_s.gsub('"', '\"') end
#autocorrected_value_for_hash(node) (private)
[ GitHub ]# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 151
def autocorrected_value_for_hash(node) hash_string = node.children.map do |child| key = autocorrected_value_in_hash(child.key) value = autocorrected_value_in_hash(child.value) "#{key}=>#{value}" end.join(', ') "{#{hash_string}}" end
#autocorrected_value_for_string(node) (private)
[ GitHub ]# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 121
def autocorrected_value_for_string(node) if node.source.start_with?("'", '%q') node.children.last.inspect[1..-2] else node.children.last end end
#autocorrected_value_for_symbol(node) (private)
[ GitHub ]# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 129
def autocorrected_value_for_symbol(node) end_pos = node.loc.end ? node.loc.end.begin_pos : node.source_range.end_pos range_between(node.loc.begin.end_pos, end_pos).source.gsub('"', '\"') end
#autocorrected_value_in_hash(node) (private)
Metrics/AbcSize
# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 162
def autocorrected_value_in_hash(node) case node.type when :int node.children.last.to_i.to_s when :float node.children.last.to_f.to_s when :str "\\\"#{node.value.to_s.gsub('"') { '\\\\\"' }}\\\"" when :sym autocorrected_value_in_hash_for_symbol(node) when :array autocorrected_value_for_array(node) when :hash autocorrected_value_for_hash(node) else node.source.gsub('"', '\"') end end
#autocorrected_value_in_hash_for_symbol(node) (private)
[ GitHub ]# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 136
def autocorrected_value_in_hash_for_symbol(node) # TODO: We need to detect symbol unacceptable names more correctly if / |"|'/.match?(node.value.to_s) ":\\\"#{node.value.to_s.gsub('"') { '\\\\\"' }}\\\"" else ":#{node.value}" end end
    #ends_heredoc_line?(node)  ⇒ Boolean  (private)
  
# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 192
def ends_heredoc_line?(node) grandparent = node.parent.parent return false unless grandparent&.dstr_type? && grandparent.heredoc? line = processed_source.lines[node.last_line - 1] line.size == node.loc.last_column + 1 end
#handle_special_regexp_chars(begin_node, value) (private)
Metrics/CyclomaticComplexity
# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 100
def handle_special_regexp_chars(begin_node, value) parent_node = begin_node.parent return value unless parent_node.regexp_type? && parent_node.slash_literal? && value['/'] # When a literal string containing a forward slash preceded by backslashes # is interpolated inside a regexp, the number of resultant backslashes in the # compiled Regexp is `(2(n+1) / 4)+1`, where `n` is the number of backslashes # inside the interpolation. # ie. 0-2 backslashes is compiled to 1, 3-6 is compiled to 3, etc. # This maintains that same behavior in order to ensure the Regexp behavior # does not change upon removing the interpolation. value.gsub(%r{(\\*)/}) do backslashes = Regexp.last_match[1] backslash_count = backslashes.length needed_backslashes = (2 * ((backslash_count + 1) / 4)) + 1 "#{'\\' * needed_backslashes}/" end end
    #in_array_percent_literal?(node)  ⇒ Boolean  (private)
  
# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 200
def in_array_percent_literal?(node) parent = node.parent return false unless parent.type?(:dstr, :dsym) grandparent = parent.parent grandparent&.array_type? && grandparent.percent_literal? end
    #offending?(node)  ⇒ Boolean  (private)
  
# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 57
def offending?(node) node && !special_keyword?(node) && prints_as_self?(node) && # Special case for `Layout/TrailingWhitespace` !(space_literal?(node) && ends_heredoc_line?(node)) && # Handled by `Lint/ArrayLiteralInRegexp` !array_in_regexp?(node) end
#on_interpolation(begin_node)
[ GitHub ]# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 28
def on_interpolation(begin_node) final_node = begin_node.children.last return unless offending?(final_node) # %W and %I split the content into words before expansion # treating each interpolation as a word component, so # interpolation should not be removed if the expanded value # contains a space character. = autocorrected_value(final_node) = handle_special_regexp_chars(begin_node, ) return if in_array_percent_literal?(begin_node) && /\s|\A\z/.match?() add_offense(final_node) do |corrector| next if final_node.dstr_type? # nested, fixed in next iteration replacement = if final_node.str_type? && !final_node.value.valid_encoding? final_node.source.delete_prefix('"').delete_suffix('"') else end corrector.replace(final_node.parent, replacement) end end
    #prints_as_self?(node)  ⇒ Boolean  (private)
  
Does node print its own source when converted to a string?
# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 183
def prints_as_self?(node) node.basic_literal? || (COMPOSITE.include?(node.type) && node.children.all? { |child| prints_as_self?(child) }) end
    #space_literal?(node)  ⇒ Boolean  (private)
  
# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 188
def space_literal?(node) node.str_type? && node.value.valid_encoding? && node.value.blank? end
    #special_keyword?(node)  ⇒ Boolean  (private)
  
# File 'lib/rubocop/cop/lint/literal_in_interpolation.rb', line 67
def special_keyword?(node) # handle strings like __FILE__ (node.str_type? && !node.loc.respond_to?(:begin)) || node.source_range.is?('__LINE__') end