Sage/dynamodb_framework

View on GitHub
lib/dynamodb_framework/dynamodb_index.rb

Summary

Maintainability
B
5 hrs
Test Coverage
B
86%
module DynamoDbFramework
  module Index

    class InvalidConfigException < StandardError
      def initialize(message)
        super(message)
      end
    end

    def config
      details = {
          index_name: full_index_name,
          table: self.instance_variable_get(:@table),
          partition_key: self.instance_variable_get(:@partition_key)
      }
      if self.instance_variable_defined?(:@range_key)
        details[:range_key] = self.instance_variable_get(:@range_key)
      end
      details
    end

    def index_name(value)
      self.instance_variable_set(:@index_name, value)
    end

    def full_index_name
      unless self.instance_variable_defined?(:@index_name)
        raise DynamoDbFramework::Index::InvalidConfigException.new('Index name must be specified.')
      end
      index_name = self.instance_variable_get(:@index_name)
      if DynamoDbFramework.namespace != nil
        index_name = "#{DynamoDbFramework.namespace}#{DynamoDbFramework.namespace_delimiter}#{index_name}"
      end
      index_name
    end

    def table(value)
      self.instance_variable_set(:@table, value)
    end

    def partition_key(field, type)
      self.instance_variable_set(:@partition_key, { field: field, type: type })
    end

    def range_key(field, type)
      self.instance_variable_set(:@range_key, { field: field, type: type })
    end

    def create(store: DynamoDbFramework.default_store, read_capacity: 25, write_capacity: 25, submit: true, billing_mode: 'PROVISIONED')
      unless self.instance_variable_defined?(:@table)
        raise DynamoDbFramework::Index::InvalidConfigException.new('Table must be specified.')
      end
      table = self.instance_variable_get(:@table)
      table_name = table.config[:table_name]

      #make method idempotent
      if exists?(store: store)
        wait_until_active(store: store)
        return
      end

      unless self.instance_variable_defined?(:@partition_key)
        raise DynamoDbFramework::Index::InvalidConfigException.new('Partition key must be specified.')
      end
      partition_key = self.instance_variable_get(:@partition_key)

      if self.instance_variable_defined?(:@range_key)
        range_key = self.instance_variable_get(:@range_key)
      end

      builder = DynamoDbFramework::AttributesBuilder.new
      builder.add({ name: partition_key[:field], type: partition_key[:type], key: :partition })
      if range_key != nil
        builder.add({ name: range_key[:field], type: range_key[:type], key: :range })
      end
      if partition_key[:field] != table.config[:partition_key][:field] && range_key != nil && range_key[:field] != table.config[:partition_key][:field]
        builder.add({ name: table.config[:partition_key][:field], type: table.config[:partition_key][:type] })
      end
      if table.config[:range_key] != nil && partition_key[:field] != table.config[:range_key][:field] && range_key != nil && range_key[:field] != table.config[:range_key][:field]
        builder.add({ name: table.config[:range_key][:field], type: table.config[:range_key][:type] })
      end

      table_manager = DynamoDbFramework::TableManager.new(store)

      range_key_field = range_key[:field] unless range_key == nil

      index =table_manager.create_global_index(full_index_name, partition_key[:field], range_key_field, read_capacity, write_capacity, billing_mode)

      if submit
        table_manager.add_index(table_name, builder.attributes, index, billing_mode)
      end

      index
    end

    def update(store: DynamoDbFramework.default_store, read_capacity:, write_capacity:)
      unless self.instance_variable_defined?(:@table)
        raise DynamoDbFramework::Index::InvalidConfigException.new('Table must be specified.')
      end
      table = self.instance_variable_get(:@table)

      DynamoDbFramework::TableManager.new(store).update_index_throughput(table.config[:table_name], full_index_name, read_capacity, write_capacity)
    end

    def drop(store: DynamoDbFramework.default_store)
      unless self.instance_variable_defined?(:@table)
        raise DynamoDbFramework::Index::InvalidConfigException.new('Table must be specified.')
      end
      table = self.instance_variable_get(:@table)
      table_name = table.config[:table_name]

      unless exists?(store: store)
        return
      end

      DynamoDbFramework::TableManager.new(store).drop_index(table_name, full_index_name)
    end

    def exists?(store: DynamoDbFramework.default_store)
      unless self.instance_variable_defined?(:@table)
        raise DynamoDbFramework::Index::InvalidConfigException.new('Table must be specified.')
      end
      table = self.instance_variable_get(:@table)
      table_name = table.config[:table_name]
      DynamoDbFramework::TableManager.new(store).has_index?(table_name, full_index_name)
    end

    def wait_until_active(store: DynamoDbFramework.default_store)
      unless self.instance_variable_defined?(:@table)
        raise DynamoDbFramework::Index::InvalidConfigException.new('Table must be specified.')
      end
      table = self.instance_variable_get(:@table)
      table_name = table.config[:table_name]
      DynamoDbFramework::TableManager.new(store).wait_until_index_active(table_name, full_index_name)
    end

    def query(partition:)
      DynamoDbFramework::Query.new(index_name: config[:index_name], table_name: config[:table].config[:table_name], partition_key: config[:partition_key][:field], partition_value: partition)
    end

  end
end