Welcome to mirror list, hosted at ThFree Co, Russian Federation.

user_interacted_project.rb « models « app - gitlab.com/gitlab-org/gitlab-foss.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: 1c615777018c5cae39d1dc4ed20a9031bcc50de0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
# frozen_string_literal: true

class UserInteractedProject < ApplicationRecord
  belongs_to :user
  belongs_to :project

  validates :project_id, presence: true
  validates :user_id, presence: true

  CACHE_EXPIRY_TIME = 1.day

  class << self
    def track(event)
      # For events without a project, we simply don't care.
      # An example of this is the creation of a snippet (which
      # is not related to any project).
      return unless event.project_id

      attributes = {
        project_id: event.project_id,
        user_id: event.author_id
      }

      cached_exists?(attributes) do
        transaction(requires_new: true) do
          where(attributes).select(1).first || create!(attributes)
          true # not caching the whole record here for now
        rescue ActiveRecord::RecordNotUnique
          # Note, above queries are not atomic and prone
          # to race conditions (similar like #find_or_create!).
          # In the case where we hit this, the record we want
          # already exists - shortcut and return.
          true
        end
      end
    end

    private

    def cached_exists?(project_id:, user_id:, &block)
      cache_key = "user_interacted_projects:#{project_id}:#{user_id}"
      Rails.cache.fetch(cache_key, expires_in: CACHE_EXPIRY_TIME, &block)
    end
  end
end