jnbt/candy_check

View on GitHub
lib/candy_check/app_store/verifier.rb

Summary

Maintainability
A
0 mins
Test Coverage
module CandyCheck
  module AppStore
    # Verifies receipts against the verification servers.
    # The call return either an {Receipt} or a {VerificationFailure}
    class Verifier
      # HTTPS endpoint for production receipts
      PRODUCTION_ENDPOINT = "https://buy.itunes.apple.com/verifyReceipt".freeze
      # HTTPS endpoint for sandbox receipts
      SANDBOX_ENDPOINT = "https://sandbox.itunes.apple.com/verifyReceipt".freeze
      # Status code from production endpoint when receiving a sandbox
      # receipt which occurs during the app's review process
      REDIRECT_TO_SANDBOX_CODE = 21_007
      # Status code from the sandbox endpoint when receiving a production
      # receipt
      REDIRECT_TO_PRODUCTION_CODE = 21_008

      # @return [Config] the current configuration
      attr_reader :config

      # Initializes a new verifier for the application which is bound
      # to a configuration
      # @param config [Config]
      def initialize(config)
        @config = config
      end

      # Calls a verification for the given input
      # @param receipt_data [String] the raw data to be verified
      # @param secret [String] the optional shared secret
      # @return [Receipt] if successful
      # @return [VerificationFailure] otherwise
      def verify(receipt_data, secret = nil)
        fetch_receipt_information(Verification, [receipt_data, secret])
      end

      # Calls a subscription verification for the given input
      # @param receipt_data [String] the raw data to be verified
      # @param secret [String] optional: shared secret
      # @param product_ids [Array<String>] optional: products to filter
      # @return [ReceiptCollection] if successful
      # @return [Verification] otherwise
      def verify_subscription(receipt_data, secret = nil, product_ids = nil)
        args = [receipt_data, secret, product_ids]
        fetch_receipt_information(SubscriptionVerification, args)
      end

      private

      def fetch_receipt_information(verifier_class, args)
        default_endpoint, opposite_endpoint = endpoints
        result = call_for(verifier_class, args.dup.unshift(default_endpoint))
        return call_for(verifier_class, args.dup.unshift(opposite_endpoint)) if should_retry?(result)

        result
      end

      def call_for(verifier_class, args)
        verifier_class.new(*args).call!
      end

      def should_retry?(result)
        result.is_a?(VerificationFailure) && redirect?(result)
      end

      def endpoints
        if config.production?
          [PRODUCTION_ENDPOINT, SANDBOX_ENDPOINT]
        else
          [SANDBOX_ENDPOINT, PRODUCTION_ENDPOINT]
        end
      end

      def redirect_code
        if config.production?
          REDIRECT_TO_SANDBOX_CODE
        else
          REDIRECT_TO_PRODUCTION_CODE
        end
      end

      def redirect?(failure)
        failure.code == redirect_code
      end
    end
  end
end