Skip to content

Commit 2ab60e3

Browse files
committed
Fix upserting with mixed data types
1 parent 96e3b04 commit 2ab60e3

File tree

5 files changed

+28
-1
lines changed

5 files changed

+28
-1
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
#### Fixed
44

55
- [#1318](https://github.com/rails-sqlserver/activerecord-sqlserver-adapter/pull/1318) Reverse order of values when upserting
6+
- [#1321](https://github.com/rails-sqlserver/activerecord-sqlserver-adapter/pull/1321) Fix SQL statement to calculate `updated_at` when upserting
67

78
## v8.0.5
89

lib/active_record/connection_adapters/sqlserver/database_statements.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -585,7 +585,7 @@ def joining_on_columns_with_uniqueness_constraints(columns_with_uniqueness_const
585585
def build_sql_for_recording_timestamps_when_updating(insert:)
586586
insert.model.timestamp_attributes_for_update_in_model.filter_map do |column_name|
587587
if insert.send(:touch_timestamp_attribute?, column_name)
588-
"target.#{quote_column_name(column_name)}=CASE WHEN (#{insert.updatable_columns.map { |column| "(COALESCE(target.#{quote_column_name(column)}, 'NULL') = COALESCE(source.#{quote_column_name(column)}, 'NULL'))" }.join(" AND ")}) THEN target.#{quote_column_name(column_name)} ELSE #{high_precision_current_timestamp} END,"
588+
"target.#{quote_column_name(column_name)}=CASE WHEN (#{insert.updatable_columns.map { |column| "(source.#{quote_column_name(column)} = target.#{quote_column_name(column)} OR (source.#{quote_column_name(column)} IS NULL AND target.#{quote_column_name(column)} IS NULL))" }.join(" AND ")}) THEN target.#{quote_column_name(column_name)} ELSE #{high_precision_current_timestamp} END,"
589589
end
590590
end.join
591591
end

test/cases/insert_all_test_sqlserver.rb

+15
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22

33
require "cases/helper_sqlserver"
44
require "models/book"
5+
require "models/sqlserver/recurring_task"
56

67
class InsertAllTestSQLServer < ActiveRecord::TestCase
78
# Test ported from the Rails `main` branch that is not on the `8-0-stable` branch.
@@ -27,4 +28,18 @@ def test_upsert_all_only_applies_last_value_when_given_duplicate_identifiers
2728
]
2829
assert_equal "expected_new_name", Book.find(112).name
2930
end
31+
32+
test "upsert_all recording of timestamps works with mixed datatypes" do
33+
task = RecurringTask.create!(
34+
key: "abcdef",
35+
priority: 5
36+
)
37+
38+
RecurringTask.upsert_all([{
39+
id: task.id,
40+
priority: nil
41+
}])
42+
43+
assert_not_equal task.updated_at, RecurringTask.find(task.id).updated_at
44+
end
3045
end
+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
class RecurringTask < ActiveRecord::Base
2+
self.table_name = "recurring_tasks"
3+
end

test/schema/sqlserver_specific_schema.rb

+8
Original file line numberDiff line numberDiff line change
@@ -360,4 +360,12 @@
360360
name varchar(255)
361361
)
362362
TABLE_IN_OTHER_SCHEMA_USED_BY_MODEL
363+
364+
create_table "recurring_tasks", force: true do |t|
365+
t.string :key
366+
t.integer :priority, default: 0
367+
368+
t.datetime2 :created_at
369+
t.datetime2 :updated_at
370+
end
363371
end

0 commit comments

Comments
 (0)