Module: ActiveRecord::ConnectionAdapters::Spanner::DatabaseStatements
- Included in:
- ActiveRecord::ConnectionAdapters::SpannerAdapter
- Defined in:
- lib/active_record/connection_adapters/spanner/database_statements.rb
Constant Summary collapse
- VERSION_7_1_0 =
Gem::Version.create "7.1.0"
- RequestOptions =
Google::Cloud::Spanner::V1::RequestOptions
- COMMENT_REGEX =
ActiveRecord::ConnectionAdapters::AbstractAdapter::COMMENT_REGEX
Instance Method Summary collapse
- #_has_pk_binding(pk, binds) ⇒ Object
- #append_request_tag_from_query_logs(sql, binds) ⇒ Object
- #append_request_tag_from_query_logs_with_format(sql, binds, prefix) ⇒ Object
- #begin_db_transaction ⇒ Object
-
#begin_isolated_db_transaction(isolation) ⇒ Object
Begins a transaction on the database with the specified isolation level.
- #commit_db_transaction ⇒ Object
- #exec_mutation(mutation) ⇒ Object
-
#exec_query(sql, name = "SQL", binds = [], prepare: false) ⇒ Object
rubocop:disable Lint/UnusedMethodArgument.
- #exec_update(sql, name = "SQL", binds = []) ⇒ Object (also: #exec_delete)
-
#execute(sql, name = nil, binds = []) ⇒ Object
DDL, DML and DQL Statements.
- #execute_ddl(statements) ⇒ Object
- #execute_query_or_dml(statement_type, sql, name, binds) ⇒ Object
- #internal_exec_query(sql, name = "SQL", binds = [], prepare: false, async: false, allow_retry: false) ⇒ Object
- #internal_execute(sql, name = "SQL", binds = [], prepare: false, async: false, allow_retry: false) ⇒ Object
-
#query(sql, name = nil) ⇒ Object
ActiveRecord.gem_version < VERSION_7_1_0.
- #rollback_db_transaction ⇒ Object
- #sql_for_insert(sql, pk, binds) ⇒ Object
-
#transaction(requires_new: nil, isolation: nil, joinable: true) ⇒ Object
Transaction.
- #transaction_isolation_levels ⇒ Object
- #truncate(table_name, name = nil) ⇒ Object
- #update(arel, name = nil, binds = []) ⇒ Object (also: #delete)
- #write_query?(sql) ⇒ Boolean
Instance Method Details
#_has_pk_binding(pk, binds) ⇒ Object
160 161 162 163 164 165 166 167 168 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 160 def _has_pk_binding pk, binds if pk.respond_to? :each has_value = true pk.each { |col| has_value &&= binds.any? { |bind| bind.name == col } } has_value else binds.any? { |bind| bind.name == pk } end end |
#append_request_tag_from_query_logs(sql, binds) ⇒ Object
85 86 87 88 89 90 91 92 93 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 85 def append_request_tag_from_query_logs sql, binds legacy_formatter_prefix = "/*request_tag:true," sql_commenter_prefix = "/*request_tag='true'," if sql.start_with? legacy_formatter_prefix append_request_tag_from_query_logs_with_format sql, binds, legacy_formatter_prefix elsif sql.start_with? sql_commenter_prefix append_request_tag_from_query_logs_with_format sql, binds, sql_commenter_prefix end end |
#append_request_tag_from_query_logs_with_format(sql, binds, prefix) ⇒ Object
95 96 97 98 99 100 101 102 103 104 105 106 107 108 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 95 def append_request_tag_from_query_logs_with_format sql, binds, prefix end_of_comment = sql.index "*/", prefix.length return unless end_of_comment request_tag = sql[prefix.length, end_of_comment - prefix.length] = binds.find { |bind| bind.is_a? RequestOptions } || RequestOptions.new if .request_tag == "" .request_tag = request_tag else .request_tag += "," + request_tag end binds.append end |
#begin_db_transaction ⇒ Object
259 260 261 262 263 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 259 def begin_db_transaction log "BEGIN" do @connection.begin_transaction end end |
#begin_isolated_db_transaction(isolation) ⇒ Object
Begins a transaction on the database with the specified isolation level. Cloud Spanner only supports isolation level :serializable, but also defines three additional 'isolation levels' that can be used to start specific types of Spanner transactions:
- :read_only: Starts a read-only snapshot transaction using a strong timestamp bound.
- :buffered_mutations: Starts a read/write transaction that will use mutations instead of DML for single-row inserts/updates/deletes. Mutations are buffered locally until the transaction is committed, and any changes during a transaction cannot be read by the application.
- :pdml: Starts a Partitioned DML transaction. Executing multiple DML statements in one PDML transaction block is NOT supported A PDML transaction is not guaranteed to be atomic. See https://cloud.google.com/spanner/docs/dml-partitioned for more information.
In addition to the above, a Hash containing read-only snapshot options may be used to start a specific read-only snapshot:
- { timestamp: Time } Starts a read-only snapshot at the given timestamp.
- { staleness: Integer } Starts a read-only snapshot with the given staleness in seconds.
- { strong:
} Starts a read-only snapshot with strong timestamp bound (this is the same as :read_only)
283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 283 def begin_isolated_db_transaction isolation if isolation.is_a? Hash raise "Unsupported isolation level: #{isolation}" unless \ isolation[:timestamp] || isolation[:staleness] || isolation[:strong] raise "Only one option is supported. It must be one of `timestamp`, `staleness` or `strong`." \ if isolation.count != 1 else raise "Unsupported isolation level: #{isolation}" unless \ [:serializable, :read_only, :buffered_mutations, :pdml].include? isolation end log "BEGIN #{isolation}" do @connection.begin_transaction isolation end end |
#commit_db_transaction ⇒ Object
299 300 301 302 303 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 299 def commit_db_transaction log "COMMIT" do @connection.commit_transaction end end |
#exec_mutation(mutation) ⇒ Object
170 171 172 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 170 def exec_mutation mutation @connection.current_transaction.buffer mutation end |
#exec_query(sql, name = "SQL", binds = [], prepare: false) ⇒ Object
rubocop:disable Lint/UnusedMethodArgument
139 140 141 142 143 144 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 139 def exec_query sql, name = "SQL", binds = [], prepare: false # rubocop:disable Lint/UnusedMethodArgument result = execute sql, name, binds ActiveRecord::Result.new( result.fields.keys.map(&:to_s), result.rows.map(&:values) ) end |
#exec_update(sql, name = "SQL", binds = []) ⇒ Object Also known as: exec_delete
188 189 190 191 192 193 194 195 196 197 198 199 200 201 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 188 def exec_update sql, name = "SQL", binds = [] result = execute sql, name, binds # Make sure that we consume the entire result stream before trying to get the stats. # This is required because the ExecuteStreamingSql RPC is also used for (Partitioned) DML, # and this RPC can return multiple partial result sets for DML as well. Only the last partial # result set will contain the statistics. Although there will never be any rows, this makes # sure that the stream is fully consumed. result.rows.each { |_| } return result.row_count if result.row_count raise ActiveRecord::StatementInvalid.new( "DML statement is invalid.", sql: sql ) end |
#execute(sql, name = nil, binds = []) ⇒ Object
DDL, DML and DQL Statements
20 21 22 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 20 def execute sql, name = nil, binds = [] internal_execute sql, name, binds end |
#execute_ddl(statements) ⇒ Object
216 217 218 219 220 221 222 223 224 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 216 def execute_ddl statements log "MIGRATION", "SCHEMA" do ActiveSupport::Dependencies.interlock.permit_concurrent_loads do @connection.execute_ddl statements end end rescue Google::Cloud::Error => error raise ActiveRecord::StatementInvalid, error end |
#execute_query_or_dml(statement_type, sql, name, binds) ⇒ Object
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 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 51 def execute_query_or_dml statement_type, sql, name, binds transaction_required = statement_type == :dml materialize_transactions # First process and remove any hints in the binds that indicate that # a different read staleness should be used than the default. staleness_hint = binds.find { |b| b.is_a? Arel::Visitors::StalenessHint } if staleness_hint selector = Google::Cloud::Spanner::Session.single_use_transaction staleness_hint.value binds.delete staleness_hint end = binds.find { |b| b.is_a? RequestOptions } if binds.delete end log_args = [sql, name] log_args.concat [binds, type_casted_binds(binds)] if log_statement_binds log(*log_args) do types, params = to_types_and_params binds ActiveSupport::Dependencies.interlock.permit_concurrent_loads do if transaction_required transaction do @connection.execute_query sql, params: params, types: types, request_options: end else @connection.execute_query sql, params: params, types: types, single_use_selector: selector, request_options: end end end end |
#internal_exec_query(sql, name = "SQL", binds = [], prepare: false, async: false, allow_retry: false) ⇒ Object
24 25 26 27 28 29 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 24 def internal_exec_query sql, name = "SQL", binds = [], prepare: false, async: false, allow_retry: false result = internal_execute sql, name, binds, prepare: prepare, async: async, allow_retry: allow_retry ActiveRecord::Result.new( result.fields.keys.map(&:to_s), result.rows.map(&:values) ) end |
#internal_execute(sql, name = "SQL", binds = [], prepare: false, async: false, allow_retry: false) ⇒ Object
31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 31 def internal_execute sql, name = "SQL", binds = [], prepare: false, async: false, allow_retry: false # rubocop:disable Lint/UnusedMethodArgument, Metrics/LineLength statement_type = sql_statement_type sql # Call `transform` to invoke any query transformers that might have been registered. sql = transform sql append_request_tag_from_query_logs sql, binds if preventing_writes? && [:dml, :ddl].include?(statement_type) raise ActiveRecord::ReadOnlyError( "Write query attempted while in readonly mode: #{sql}" ) end if statement_type == :ddl execute_ddl sql else execute_query_or_dml statement_type, sql, name, binds end end |
#query(sql, name = nil) ⇒ Object
ActiveRecord.gem_version < VERSION_7_1_0
135 136 137 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 135 def query sql, name = nil exec_query sql, name end |
#rollback_db_transaction ⇒ Object
305 306 307 308 309 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 305 def rollback_db_transaction log "ROLLBACK" do @connection.rollback_transaction end end |
#sql_for_insert(sql, pk, binds) ⇒ Object
113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 113 def sql_for_insert sql, pk, binds, returning if pk && !_has_pk_binding(pk, binds) # Add the primary key to the columns that should be returned if there is no value specified for it. returning ||= [] returning |= if pk.respond_to? :each pk else [pk] end end if returning&.any? returning_columns_statement = returning.map { |c| quote_column_name c }.join(", ") sql = "#{sql} THEN RETURN #{returning_columns_statement}" end [sql, binds] end |
#transaction(requires_new: nil, isolation: nil, joinable: true) ⇒ Object
Transaction
228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 228 def transaction requires_new: nil, isolation: nil, joinable: true if !requires_new && current_transaction.joinable? return super end backoff = 0.2 begin super rescue ActiveRecord::StatementInvalid => err if err.cause.is_a? Google::Cloud::AbortedError sleep(delay_from_aborted(err) || backoff *= 1.3) retry end raise end end |
#transaction_isolation_levels ⇒ Object
245 246 247 248 249 250 251 252 253 254 255 256 257 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 245 def transaction_isolation_levels { read_uncommitted: "READ UNCOMMITTED", read_committed: "READ COMMITTED", repeatable_read: "REPEATABLE READ", serializable: "SERIALIZABLE", # These are not really isolation levels, but it is the only (best) way to pass in additional # transaction options to the connection. read_only: "READ_ONLY", buffered_mutations: "BUFFERED_MUTATIONS" } end |
#truncate(table_name, name = nil) ⇒ Object
204 205 206 207 208 209 210 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 204 def truncate table_name, name = nil Array(table_name).each do |t| log "TRUNCATE #{t}", name do @connection.truncate t end end end |
#update(arel, name = nil, binds = []) ⇒ Object Also known as: delete
174 175 176 177 178 179 180 181 182 183 184 185 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 174 def update arel, name = nil, binds = [] # Add a `WHERE TRUE` if it is an update_all or delete_all call that uses DML. if !should_use_mutation(arel) && arel.respond_to?(:ast) && arel.ast.wheres.empty? arel.ast.wheres << Arel::Nodes::SqlLiteral.new("TRUE") end return super unless should_use_mutation arel raise "Unsupported update for use with mutations: #{arel}" unless arel.is_a? Arel::DeleteManager exec_mutation create_delete_all_mutation arel if arel.is_a? Arel::DeleteManager 0 # Affected rows (unknown) end |
#write_query?(sql) ⇒ Boolean
212 213 214 |
# File 'lib/active_record/connection_adapters/spanner/database_statements.rb', line 212 def write_query? sql sql_statement_type(sql) == :dml end |