8000 Empty record when conditional finds none · Issue #128 · jesjos/active_record_upsert · GitHub
[go: up one dir, main page]
More Web Proxy on the site http://driver.im/
Skip to content
Empty record when conditional finds none #128
Open
@hrdwdmrbl

Description

@hrdwdmrbl

Context: The problem is basically https://stackoverflow.com/questions/34708509/how-to-use-returning-with-on-conflict-in-postgresql/37543015

Problem: If you have a conditional upsert, record.upsert!({arel_condition: upsert_condition}), and the condition returns no rows, then the record object will be empty.

My situation: In my case, my condition is about timestamps. self.class.arel_table[:shopify_updated_at].lteq(send(:shopify_updated_at)) where Shopify is an external source of data for my app. I only want to upsert if the current record is newer (has a later updated_at).

Proposed solution: There are solutions listed in the Stack Overflow discussion. I might suggest a simple solution which is to follow up the INSERT with a subsequent read when nothing is returned. The read should also use the upsert_keys if given. This is necessary because otherwise it is not know how to find a matching record.

For me, the upsert_keys is usually an external ID column. Something like shopify_order_id. I don't want duplicate orders, so I upsert on conflict of duplicate shopify_order_id. If a conflict is found, and the shopify_updated_at of what I'm trying to upsert is older, then by the way Postgres works, no row will be returned! So the only thing to do is to either remove the condition, modify the INSERT query, or fetch again when empty.

For example purposes only:

    def _upsert_record(upsert_attribute_names = changed, arel_condition = nil, opts = {})
        existing_attribute_names = attributes_for_create(attributes.keys)
        existing_attributes = attributes_with_values(existing_attribute_names)
        values = self.class._upsert_record(existing_attributes, upsert_attribute_names, [arel_condition].compact, opts)

        values = if values.first.to_h.empty? && opts[:upsert_keys].present?
          find_by_conditions = opts[:upsert_keys].each_with_object({}) do |upsert_key, unique_attributes|
            unique_attributes[upsert_key] = send(upsert_key)
          end
          existing_record = self.class.find_by(find_by_conditions)
          existing_record.attributes
        else
          values.first.to_h
        end

        @attributes = self.class.attributes_builder.build_from_database(values)
        @new_record = false
        changes_applied
        values
      end

Metadata

Metadata

Assignees

No one assigned

    Labels

    No labels
    No labels

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions

      0