module Repor
  module Dimensions
    class BaseDimension
      attr_reader :name, :report, :opts

      def initialize(name, report, opts={})
        @name = name
        @report = report
        @opts = opts
        validate_params!
      end

      def expression
        opts.fetch(:expression, "#{report.table_name}.#{name}")
      end

      # Do any joins/selects necessary to filter or group the relation.
      def relate(relation)
        opts.fetch(:relation, ->(r) { r }).call(relation)
      end

      # Filter the relation based on any constraints in the params
      def filter(relation)
        raise NotImplementedError
      end

      # Group the relation by the expression -- ensure this is ordered, too.
      def group(relation)
        raise NotImplementedError
      end

      # Return an ordered array of all values that should appear in
      # `Report#data`
      def group_values
        raise NotImplementedError
      end

      # Given a single (hashified) row of the SQL result, return the Ruby
      # object representing this dimension's value
      def extract_value(row)
        sanitize(row[sql_value_name])
      end

      def filter_values
        array_param(:only).uniq
      end

      # Return whether the report should filter by this dimension
      def filtering?
        filter_values.present?
      end

      def grouping?
        report.groupers.include?(self)
      end

      def order_expression
        sql_value_name
      end

      def order(relation)
        relation.order("#{order_expression} #{sort_order} #{null_order}")
      end

      def sort_order
        dimension_or_root_param(:sort_desc) ? 'DESC' : 'ASC'
      end

      def null_order
        return unless Repor.database_type == :postgres
        dimension_or_root_param(:nulls_last) ? 'NULLS LAST' : 'NULLS FIRST'
      end

      def params
        report.params.fetch(:dimensions, {}).fetch(name, {})
      end

      private

      def validate_params!
      end

      def invalid_param!(param_key, message)
        raise InvalidParamsError, "Invalid value for params[:dimensions]" \
          "[:#{name}][:#{param_key}]: #{message}"
      end

      def sql_value_name
        "_repor_dimension_#{name}"
      end

      def sanitize(raw_value)
        raw_value
      end

      def dimension_or_root_param(key)
        params.fetch(key, report.params[key])
      end

      def array_param(key)
        params.key?(key) ? Array.wrap(params[key]) : []
      end
    end
  end
end