Module: Torque::PostgreSQL::Adapter::DatabaseStatements

Included in:
Torque::PostgreSQL::Adapter
Defined in:
lib/torque/postgresql/adapter/database_statements.rb

Constant Summary collapse

EXTENDED_DATABASE_TYPES =
i[enum enum_set interval]

Instance Method Summary collapse

Instance Method Details

#column_definitions(table_name) ⇒ Object

Get the list of columns, and their definition, but only from the actual table, does not include columns that comes from inherited table



167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 167

def column_definitions(table_name)
  query("      SELECT a.attname, format_type(a.atttypid, a.atttypmod),\n             pg_get_expr(d.adbin, d.adrelid), a.attnotnull, a.atttypid, a.atttypmod,\n             c.collname, col_description(a.attrelid, a.attnum) AS comment,\n             \#{supports_identity_columns? ? 'attidentity' : quote('')} AS identity,\n             \#{supports_virtual_columns? ? 'attgenerated' : quote('')} as attgenerated\n        FROM pg_attribute a\n        LEFT JOIN pg_attrdef d ON a.attrelid = d.adrelid AND a.attnum = d.adnum\n        LEFT JOIN pg_type t ON a.atttypid = t.oid\n        LEFT JOIN pg_collation c ON a.attcollation = c.oid AND a.attcollation <> t.typcollation\n       WHERE a.attrelid = \#{quote(quote_table_name(table_name))}::regclass\n         AND a.attnum > 0 AND NOT a.attisdropped\n         \#{'AND a.attislocal' if @_dump_mode}\n       ORDER BY a.attnum\n  SQL\nend\n", "SCHEMA")

#dump_mode!Object

Switch between dump mode or not



11
12
13
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 11

def dump_mode!
  @_dump_mode = !!!@_dump_mode
end

#extended_typesObject

Get the list of extended types



41
42
43
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 41

def extended_types
  EXTENDED_DATABASE_TYPES
end

#filter_by_schemaObject

Build the condition for filtering by schema



220
221
222
223
224
225
226
227
228
229
230
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 220

def filter_by_schema
  conditions = []
  conditions << "    nspname NOT LIKE ALL (ARRAY['\#{schemas_blacklist.join(\"', '\")}'])\n  SQL\n\n  conditions << <<-SQL.squish if schemas_whitelist.any?\n    nspname LIKE ANY (ARRAY['\#{schemas_whitelist.join(\"', '\")}'])\n  SQL\n  conditions\nend\n".squish if schemas_blacklist.any?

#inherited_tablesObject

Get the list of inherited tables associated with their parent tables



135
136
137
138
139
140
141
142
143
144
145
146
147
148
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 135

def inherited_tables
  tables = query("    SELECT inhrelid::regclass  AS table_name,\n           inhparent::regclass AS inheritances\n    FROM pg_inherits\n    JOIN pg_class parent ON pg_inherits.inhparent = parent.oid\n    JOIN pg_class child  ON pg_inherits.inhrelid  = child.oid\n    ORDER BY inhrelid\n  SQL\n\n  tables.each_with_object({}) do |(child, parent), result|\n    (result[child] ||= []) << parent\n  end\nend\n", 'SCHEMA')

#initialize_type_map(m = type_map) ⇒ Object

Change some of the types being mapped



63
64
65
66
67
68
69
70
71
72
73
74
75
76
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 63

def initialize_type_map(m = type_map)
  super

  if PostgreSQL.config.geometry.enabled
    m.register_type 'box',      OID::Box.new
    m.register_type 'circle',   OID::Circle.new
    m.register_type 'line',     OID::Line.new
    m.register_type 'segment',  OID::Segment.new
  end

  if PostgreSQL.config.interval.enabled
    m.register_type 'interval', OID::Interval.new
  end
end

#list_versioned_commands(type) ⇒ Object

Get all possible schema entries that can be created via versioned commands of the provided type. Mostly for covering removals and not dump them



188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 188

def list_versioned_commands(type)
  query =
    case type
    when :function
      "        SELECT n.nspname AS schema, p.proname AS name\n        FROM pg_catalog.pg_proc p\n        INNER JOIN pg_namespace n ON n.oid = p.pronamespace\n        WHERE 1=1 AND \#{filter_by_schema.join(' AND ')};\n      SQL\n    when :type\n      <<-SQL.squish\n        SELECT n.nspname AS schema, t.typname AS name\n        FROM pg_type t\n        INNER JOIN pg_namespace n ON n.oid = t.typnamespace\n        WHERE 1=1 AND t.typtype NOT IN ('e')\n          AND \#{filter_by_schema.join(' AND ')};\n      SQL\n    when :view\n      <<-SQL.squish\n        SELECT n.nspname AS schema, c.relname AS name\n        FROM pg_class c\n        INNER JOIN pg_namespace n ON n.oid = c.relnamespace\n        WHERE 1=1 AND c.relkind IN ('v', 'm')\n          AND \#{filter_by_schema.join(' AND ')};\n      SQL\n    end\n\n  select_rows(query, 'SCHEMA')\nend\n".squish

#load_additional_types(oids = nil) ⇒ Object

:nodoc:



79
80
81
82
83
84
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 79

def load_additional_types(oids = nil)
  type_map.alias_type 'regclass', 'varchar'
  type_map.alias_type 'regconfig', 'varchar'
  super
  torque_load_additional_types(oids)
end

#schema_exists?(name, filtered: true) ⇒ Boolean

Checks if a given schema exists in the database. If filtered is given as false, then it will check regardless of whitelist and blacklist



48
49
50
51
52
53
54
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 48

def schema_exists?(name, filtered: true)
  return user_defined_schemas.include?(name.to_s) if filtered

  query_value("    SELECT 1 FROM pg_catalog.pg_namespace WHERE nspname = \#{quote(name)}\n  SQL\nend\n", "SCHEMA") == 1

#schemas_blacklistObject

List of schemas blocked by the application in the current connection



16
17
18
19
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 16

def schemas_blacklist
  @schemas_blacklist ||= Torque::PostgreSQL.config.schemas.blacklist +
    (@config.dig(:schemas, 'blacklist') || [])
end

#schemas_search_path_sanitizedObject

A list of schemas on the search path sanitized



28
29
30
31
32
33
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 28

def schemas_search_path_sanitized
  @schemas_search_path_sanitized ||= begin
    db_user = @config[:username] || ENV['USER'] || ENV['USERNAME']
    schema_search_path.split(',').map { |item| item.strip.sub('"$user"', db_user) }
  end
end

#schemas_whitelistObject

List of schemas used by the application in the current connection



22
23
24
25
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 22

def schemas_whitelist
  @schemas_whitelist ||= Torque::PostgreSQL.config.schemas.whitelist +
    (@config.dig(:schemas, 'whitelist') || [])
end

#torque_load_additional_types(oids = nil) ⇒ Object

Add the composite types to be loaded too.



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
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 87

def torque_load_additional_types(oids = nil)
  return unless torque_load_additional_types?

  # Types: (b)ase, (c)omposite, (d)omain, (e)num, (p)seudotype, (r)ange
  # (m)ultirange

  query = "    SELECT t.oid, t.typname, t.typelem, t.typdelim, t.typinput,\n           r.rngsubtype, t.typtype, t.typbasetype, t.typarray\n    FROM pg_type as t\n    LEFT JOIN pg_range as r ON oid = rngtypid\n    LEFT JOIN pg_catalog.pg_namespace n ON n.oid = t.typnamespace\n    WHERE n.nspname NOT IN ('pg_catalog', 'information_schema')\n  SQL\n\n  if oids\n    query += \" AND t.oid IN (%s)\" % oids.join(\", \")\n  else\n    query += \" AND t.typtype IN ('e')\"\n  end\n\n  options = { allow_retry: true, materialize_transactions: false }\n  internal_execute(query, 'SCHEMA', **options).each do |row|\n    if row['typtype'] == 'e' && PostgreSQL.config.enum.enabled\n      OID::Enum.create(row, type_map)\n    end\n  end\nend\n"

#torque_load_additional_types?Boolean



116
117
118
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 116

def torque_load_additional_types?
  PostgreSQL.config.enum.enabled
end

#type_exists?(name) ⇒ Boolean Also known as: data_type_exists?

Returns true if type exists.



57
58
59
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 57

def type_exists?(name)
  user_defined_types.key? name.to_s
end

#user_defined_schemasObject

Get the list of schemas that were created by the user



151
152
153
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 151

def user_defined_schemas
  query_values(user_defined_schemas_sql, 'SCHEMA')
end

#user_defined_schemas_sqlObject

Build the query for allowed schemas



156
157
158
159
160
161
162
163
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 156

def user_defined_schemas_sql
  "    SELECT nspname\n    FROM pg_catalog.pg_namespace\n    WHERE 1=1 AND \#{filter_by_schema.join(' AND ')}\n    ORDER BY oid\n  SQL\nend\n".squish

#user_defined_types(*categories) ⇒ Object

Gets a list of user defined types. You can even choose the category filter



122
123
124
125
126
127
128
129
130
131
132
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 122

def user_defined_types(*categories)
  categories = categories.compact.presence || %w[c e p r m]

  query("    SELECT t.typname, t.typtype\n    FROM pg_type as t\n    LEFT JOIN pg_catalog.pg_namespace n ON n.oid = t.typnamespace\n    WHERE n.nspname NOT IN ('pg_catalog', 'information_schema')\n    AND t.typtype IN ('\#{categories.join(\"', '\")}')\n  SQL\nend\n", 'SCHEMA').to_h

#valid_type?(type) ⇒ Boolean

Check if a given type is valid.



36
37
38
# File 'lib/torque/postgresql/adapter/database_statements.rb', line 36

def valid_type?(type)
  super || extended_types.include?(type)
end