# frozen_string_literal: true require_relative '../mixins/packer' module Trifle module Stats module Driver class Mongo include Mixins::Packer attr_accessor :client, :collection_name, :separator def initialize(client, collection_name: 'trifle_stats') @client = client @collection_name = collection_name @separator = '::' end def self.setup!(client, collection_name: 'trifle_stats') client[collection_name].create client[collection_name].indexes.create_one({ key: 1 }, unique: true) end def inc(keys:, **values) data = self.class.pack(hash: { data: values }) collection.bulk_write( keys.map do |key| upsert_operation('$inc', pkey: key.join(separator), data: data) end ) end def set(keys:, **values) data = self.class.pack(hash: { data: values }) collection.bulk_write( keys.map do |key| upsert_operation('$set', pkey: key.join(separator), data: data) end ) end def upsert_operation(operation, pkey:, data:) { update_many: { filter: { key: pkey }, update: { operation => data }, upsert: true } } end def get(keys:) pkeys = keys.map { |key| key.join(separator) } data = collection.find(key: { '$in' => pkeys }) map = data.inject({}) { |o, d| o.merge(d['key'] => d['data']) } pkeys.map { |pkey| map[pkey] || {} } end private def collection client[collection_name] end end end end end