aboutsummaryrefslogtreecommitdiffstats
path: root/activesupport/lib/active_support/json.rb
blob: d1203bd21bdb18e96d65f7e5928fe58bebf72e2e (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
require 'active_support/json/encoders'

module ActiveSupport
  module JSON #:nodoc:
    class CircularReferenceError < StandardError #:nodoc:
    end
    
    # A string that returns itself as as its JSON-encoded form.
    class Variable < String #:nodoc:
      def to_json
        self
      end
    end
    
    # When +true+, Hash#to_json will omit quoting string or symbol keys
    # if the keys are valid JavaScript identifiers.  Note that this is
    # technically improper JSON (all object keys must be quoted), so if
    # you need strict JSON compliance, set this option to +false+.
    mattr_accessor :unquote_hash_key_identifiers
    @@unquote_hash_key_identifiers = true

    class << self
      REFERENCE_STACK_VARIABLE = :json_reference_stack
      
      def encode(value)
        raise_on_circular_reference(value) do
          Encoders[value.class].call(value)
        end
      end
      
      def can_unquote_identifier?(key)
        return false unless unquote_hash_key_identifiers
        key.to_s =~ /^[[:alpha:]_$][[:alnum:]_$]*$/
      end
      
      protected
        def raise_on_circular_reference(value)
          stack = Thread.current[REFERENCE_STACK_VARIABLE] ||= []
          raise CircularReferenceError, 'object references itself' if
            stack.include? value
          stack << value
          yield
        ensure
          stack.pop
        end
    end
  end
end