Module: RubyPgExtras

Defined in:
lib/ruby-pg-extras.rb,
lib/ruby_pg_extras/version.rb,
lib/ruby_pg_extras/index_info.rb,
lib/ruby_pg_extras/table_info.rb,
lib/ruby_pg_extras/size_parser.rb,
lib/ruby_pg_extras/diagnose_data.rb,
lib/ruby_pg_extras/diagnose_print.rb,
lib/ruby_pg_extras/index_info_print.rb,
lib/ruby_pg_extras/table_info_print.rb

Defined Under Namespace

Classes: DiagnoseData, DiagnosePrint, IndexInfo, IndexInfoPrint, SizeParser, TableInfo, TableInfoPrint

Constant Summary collapse

NEW_PG_STAT_STATEMENTS =
"1.8"
PG_STAT_STATEMENTS_17 =
"1.11"
QUERIES =
%i(
  add_extensions bloat blocking cache_hit db_settings
  calls extensions table_cache_hit tables index_cache_hit
  indexes index_size index_usage index_scans null_indexes locks all_locks
  long_running_queries mandelbrot outliers
  records_rank seq_scans table_index_scans table_indexes_size
  table_size total_index_size total_table_size
  unused_indexes duplicate_indexes vacuum_stats kill_all kill_pid
  pg_stat_statements_reset buffercache_stats
  buffercache_usage ssl_used connections
)
DEFAULT_SCHEMA =
ENV["PG_EXTRAS_SCHEMA"] || "public"
DEFAULT_ARGS =
Hash.new({}).merge({
  calls: { limit: 10 },
  calls_legacy: { limit: 10 },
  calls_17: { limit: 10 },
  long_running_queries: { threshold: "500 milliseconds" },
  locks: { limit: 20 },
  blocking: { limit: 20 },
  outliers: { limit: 10 },
  outliers_legacy: { limit: 10 },
  outliers_17: { limit: 10 },
  buffercache_stats: { limit: 10 },
  buffercache_usage: { limit: 20 },
  unused_indexes: { max_scans: 50, schema: DEFAULT_SCHEMA },
  null_indexes: { min_relation_size_mb: 10 },
  index_usage: { schema: DEFAULT_SCHEMA },
  index_cache_hit: { schema: DEFAULT_SCHEMA },
  table_cache_hit: { schema: DEFAULT_SCHEMA },
  index_scans: { schema: DEFAULT_SCHEMA },
  cache_hit: { schema: DEFAULT_SCHEMA },
  seq_scans: { schema: DEFAULT_SCHEMA },
  table_index_scans: { schema: DEFAULT_SCHEMA },
  records_rank: { schema: DEFAULT_SCHEMA },
  tables: { schema: DEFAULT_SCHEMA },
  kill_pid: { pid: 0 },
})
VERSION =
"5.5.1"
@@database_url =
nil

Class Method Summary collapse

Class Method Details

.connectionObject



181
182
183
# File 'lib/ruby-pg-extras.rb', line 181

def self.connection
  @_connection ||= PG.connect(database_url)
end

.database_urlObject



189
190
191
# File 'lib/ruby-pg-extras.rb', line 189

def self.database_url
  @@database_url || ENV["RUBY_PG_EXTRAS_DATABASE_URL"] || ENV.fetch("DATABASE_URL")
end

.database_url=(value) ⇒ Object



185
186
187
# File 'lib/ruby-pg-extras.rb', line 185

def self.database_url=(value)
  @@database_url = value
end

.description_for(query_name:) ⇒ Object



163
164
165
166
167
168
169
# File 'lib/ruby-pg-extras.rb', line 163

def self.description_for(query_name:)
  first_line = File.open(
    sql_path_for(query_name: query_name)
  ) { |f| f.readline }

  first_line[/\/\*(.*?)\*\//m, 1].strip
end

.diagnose(in_format: :display_table) ⇒ Object



96
97
98
99
100
101
102
103
104
105
106
107
108
# File 'lib/ruby-pg-extras.rb', line 96

def self.diagnose(in_format: :display_table)
  data = RubyPgExtras::DiagnoseData.call

  if in_format == :display_table
    RubyPgExtras::DiagnosePrint.call(data)
  elsif in_format == :hash
    data
  elsif in_format == :array
    data.map(&:values)
  else
    raise "Invalid 'in_format' argument!"
  end
end

.display_result(result, title:, in_format:) ⇒ Object



138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
# File 'lib/ruby-pg-extras.rb', line 138

def self.display_result(result, title:, in_format:)
  case in_format
  when :array
    result.values
  when :hash
    result.to_a
  when :raw
    result
  when :display_table
    headings = if result.count > 0
        result[0].keys
      else
        ["No results"]
      end

    puts Terminal::Table.new(
      title: title,
      headings: headings,
      rows: result.values,
    )
  else
    raise "Invalid in_format option"
  end
end

.index_info(args: {}, in_format: :display_table) ⇒ Object



110
111
112
113
114
115
116
117
118
119
120
121
122
# File 'lib/ruby-pg-extras.rb', line 110

def self.index_info(args: {}, in_format: :display_table)
  data = RubyPgExtras::IndexInfo.call(args[:table_name])

  if in_format == :display_table
    RubyPgExtras::IndexInfoPrint.call(data)
  elsif in_format == :hash
    data
  elsif in_format == :array
    data.map(&:values)
  else
    raise "Invalid 'in_format' argument!"
  end
end

.run_query(query_name:, in_format:, args: {}) ⇒ Object



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
# File 'lib/ruby-pg-extras.rb', line 69

def self.run_query(query_name:, in_format:, args: {})
  if %i(calls outliers).include?(query_name)
    pg_stat_statements_ver = RubyPgExtras.connection.exec("select installed_version from pg_available_extensions where name='pg_stat_statements'")
      .to_a[0].fetch("installed_version", nil)
    if pg_stat_statements_ver != nil
      if Gem::Version.new(pg_stat_statements_ver) < Gem::Version.new(NEW_PG_STAT_STATEMENTS)
        query_name = "#{query_name}_legacy".to_sym
      elsif Gem::Version.new(pg_stat_statements_ver) >= Gem::Version.new(PG_STAT_STATEMENTS_17)
        query_name = "#{query_name}_17".to_sym
      end
    end
  end

  sql = if (custom_args = DEFAULT_ARGS[query_name].merge(args)) != {}
      sql_for(query_name: query_name) % custom_args
    else
      sql_for(query_name: query_name)
    end
  result = connection.exec(sql)

  display_result(
    result,
    title: description_for(query_name: query_name),
    in_format: in_format,
  )
end

.sql_for(query_name:) ⇒ Object



171
172
173
174
175
# File 'lib/ruby-pg-extras.rb', line 171

def self.sql_for(query_name:)
  File.read(
    sql_path_for(query_name: query_name)
  )
end

.sql_path_for(query_name:) ⇒ Object



177
178
179
# File 'lib/ruby-pg-extras.rb', line 177

def self.sql_path_for(query_name:)
  File.join(File.dirname(__FILE__), "/ruby_pg_extras/queries/#{query_name}.sql")
end

.table_info(args: {}, in_format: :display_table) ⇒ Object



124
125
126
127
128
129
130
131
132
133
134
135
136
# File 'lib/ruby-pg-extras.rb', line 124

def self.table_info(args: {}, in_format: :display_table)
  data = RubyPgExtras::TableInfo.call(args[:table_name])

  if in_format == :display_table
    RubyPgExtras::TableInfoPrint.call(data)
  elsif in_format == :hash
    data
  elsif in_format == :array
    data.map(&:values)
  else
    raise "Invalid 'in_format' argument!"
  end
end