summaryrefslogtreecommitdiff
path: root/app/models/ci/build_trace_chunks/redis.rb
blob: 003ec10789548c697e1240de884e58cb55cdaa52 (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
46
47
48
49
50
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
84
85
86
87
88
89
90
91
92
93
94
# frozen_string_literal: true

module Ci
  module BuildTraceChunks
    class Redis
      CHUNK_REDIS_TTL = 1.week
      LUA_APPEND_CHUNK = <<~EOS
        local key, new_data, offset = KEYS[1], ARGV[1], ARGV[2]
        local length = new_data:len()
        local expire = #{CHUNK_REDIS_TTL.seconds}
        local current_size = redis.call("strlen", key)
        offset = tonumber(offset)

        if offset == 0 then
          -- overwrite everything
          redis.call("set", key, new_data, "ex", expire)
          return redis.call("strlen", key)
        elseif offset > current_size then
          -- offset range violation
          return -1
        elseif offset + length >= current_size then
          -- efficiently append or overwrite and append
          redis.call("expire", key, expire)
          return redis.call("setrange", key, offset, new_data)
        else
          -- append and truncate
          local current_data = redis.call("get", key)
          new_data = current_data:sub(1, offset) .. new_data
          redis.call("set", key, new_data, "ex", expire)
          return redis.call("strlen", key)
        end
      EOS

      def available?
        true
      end

      def data(model)
        Gitlab::Redis::SharedState.with do |redis|
          redis.get(key(model))
        end
      end

      def set_data(model, new_data)
        Gitlab::Redis::SharedState.with do |redis|
          redis.set(key(model), new_data, ex: CHUNK_REDIS_TTL)
        end
      end

      def append_data(model, new_data, offset)
        Gitlab::Redis::SharedState.with do |redis|
          redis.eval(LUA_APPEND_CHUNK, keys: [key(model)], argv: [new_data, offset])
        end
      end

      def size(model)
        Gitlab::Redis::SharedState.with do |redis|
          redis.strlen(key(model))
        end
      end

      def delete_data(model)
        delete_keys([[model.build_id, model.chunk_index]])
      end

      def keys(relation)
        relation.pluck(:build_id, :chunk_index)
      end

      def delete_keys(keys)
        return if keys.empty?

        keys = keys.map { |key| key_raw(*key) }

        Gitlab::Redis::SharedState.with do |redis|
          # https://gitlab.com/gitlab-org/gitlab/-/issues/224171
          Gitlab::Instrumentation::RedisClusterValidator.allow_cross_slot_commands do
            redis.del(keys)
          end
        end
      end

      private

      def key(model)
        key_raw(model.build_id, model.chunk_index)
      end

      def key_raw(build_id, chunk_index)
        "gitlab:ci:trace:#{build_id.to_i}:chunks:#{chunk_index.to_i}"
      end
    end
  end
end