summaryrefslogtreecommitdiff
path: root/lib/pry/commands/ls.rb
blob: 1a03db15e30cbc247e1a413707c781b2e6b2ae47 (plain)
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
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
# frozen_string_literal: true

class Pry
  class Command
    class Ls < Pry::ClassCommand
      DEFAULT_OPTIONS = {
        heading_color: :bright_blue,
        public_method_color: :default,
        private_method_color: :blue,
        protected_method_color: :blue,
        method_missing_color: :bright_red,
        local_var_color: :yellow,
        pry_var_color: :default, # e.g. _, pry_instance, _file_
        instance_var_color: :blue, # e.g. @foo
        class_var_color: :bright_blue, # e.g. @@foo
        global_var_color: :default, # e.g. $CODERAY_DEBUG, $eventmachine_library
        builtin_global_color: :cyan, # e.g. $stdin, $-w, $PID
        pseudo_global_color: :cyan, # e.g. $~, $1..$9, $LAST_MATCH_INFO
        constant_color: :default, # e.g. VERSION, ARGF
        class_constant_color: :blue, # e.g. Object, Kernel
        exception_constant_color: :magenta, # e.g. Exception, RuntimeError
        unloaded_constant_color: :yellow, # Any constant that is still in .autoload? state
        separator: "  ",
        ceiling: [Object, Module, Class]
      }.freeze

      match 'ls'
      group 'Context'
      description 'Show the list of vars and methods in the current scope.'
      command_options shellwords: false, interpolate: false

      banner <<-'BANNER'
        Usage: ls [-m|-M|-p|-pM] [-q|-v] [-c|-i] [Object]
               ls [-g] [-l]

        ls shows you which methods, constants and variables are accessible to Pry. By
        default it shows you the local variables defined in the current shell, and any
        public methods or instance variables defined on the current object.

        The colours used are configurable using Pry.config.ls.*_color, and the separator
        is Pry.config.ls.separator.

        Pry.config.ls.ceiling is used to hide methods defined higher up in the
        inheritance chain, this is by default set to [Object, Module, Class] so that
        methods defined on all Objects are omitted. The -v flag can be used to ignore
        this setting and show all methods, while the -q can be used to set the ceiling
        much lower and show only methods defined on the object or its direct class.

        Also check out `find-method` command (run `help find-method`).
      BANNER

      def options(opt)
        opt.on :m, :methods, "Show public methods defined on the Object"
        opt.on :M, "instance-methods", "Show public methods defined in a " \
                                       "Module or Class"
        opt.on :p, :ppp, "Show public, protected (in yellow) and private " \
                         "(in green) methods"
        opt.on :q, :quiet, "Show only methods defined on object.singleton_class " \
                           "and object.class"
        opt.on :v, :verbose, "Show methods and constants on all super-classes " \
                             "(ignores Pry.config.ls.ceiling)"
        opt.on :g, :globals, "Show global variables, including those builtin to " \
                             "Ruby (in cyan)"
        opt.on :l, :locals, "Show hash of local vars, sorted by descending size"
        opt.on :c, :constants, "Show constants, highlighting classes (in blue), " \
                               "and exceptions (in purple).\n" \
                               "#{' ' * 32}Constants that are pending autoload? " \
                               "are also shown (in yellow)"
        opt.on :i, :ivars, "Show instance variables (in blue) and class " \
                           "variables (in bright blue)"
        opt.on :G, :grep, "Filter output by regular expression", argument: true

        if Object.respond_to?(:deprecate_constant)
          opt.on :d, :dconstants, "Show deprecated constants"
        end

        return unless Helpers::Platform.jruby?

        opt.on :J, "all-java", "Show all the aliases for methods from java " \
                               "(default is to show only prettiest)"
      end

      # Exclude -q, -v and --grep because they,
      # don't specify what the user wants to see.
      def no_user_opts?
        !(opts[:methods] || opts['instance-methods'] || opts[:ppp] ||
          opts[:globals] || opts[:locals] || opts[:constants] || opts[:ivars])
      end

      def process
        @interrogatee = args.empty? ? target_self : target.eval(args.join(' '))
        raise_errors_if_arguments_are_weird
        ls_entity = LsEntity.new(
          interrogatee: @interrogatee,
          no_user_opts: no_user_opts?,
          opts: opts,
          args: args,
          pry_instance: pry_instance
        )

        pry_instance.pager.page ls_entity.entities_table
      end

      private

      def error_list
        any_args = args.any?
        # rubocop:disable Style/CaseEquality
        non_mod_interrogatee = !(Module === @interrogatee)
        # rubocop:enable Style/CaseEquality
        [
          ['-l does not make sense with a specified Object', :locals, any_args],
          ['-g does not make sense with a specified Object', :globals, any_args],
          ['-q does not make sense with -v', :quiet, opts.present?(:verbose)],
          [
            '-M only makes sense with a Module or a Class', 'instance-methods',
            non_mod_interrogatee
          ],
          [
            '-c only makes sense with a Module or a Class', :constants,
            any_args && non_mod_interrogatee
          ]
        ]
      end

      def raise_errors_if_arguments_are_weird
        error_list.each do |message, option, invalid_expr|
          raise Pry::CommandError, message if opts.present?(option) && invalid_expr
        end
      end
    end

    Pry::Commands.add_command(Pry::Command::Ls)
  end
end