rapid7/metasploit-framework

View on GitHub
lib/rex/proto/kerberos/model/pre_auth_data_entry.rb

Summary

Maintainability
A
25 mins
Test Coverage
# -*- coding: binary -*-

module Rex
  module Proto
    module Kerberos
      module Model
        # This class provides a representation for Kerberos pre authenticated
        # data entry.
        class PreAuthDataEntry < Element

          # @!attribute type
          #   @return [Integer] The padata type
          attr_accessor :type
          # @!attribute value
          #   @return [String] The padata value, encoded
          attr_accessor :value

          # Decodes a Rex::Proto::Kerberos::Model::PreAuthDataEntry
          #
          # @param input [String, OpenSSL::ASN1::Sequence] the input to decode from
          # @return [self] if decoding succeeds
          # @raise [Rex::Proto::Kerberos::Model::Error::KerberosDecodingError] if decoding doesn't succeed
          def decode(input)
            case input
            when String
              decode_string(input)
            when OpenSSL::ASN1::ASN1Data
              decode_asn1(input)
            when OpenSSL::ASN1::Sequence
              decode_asn1(input)
            else
              raise ::Rex::Proto::Kerberos::Model::Error::KerberosDecodingError, 'Failed to decode PreAuthDataEntry, invalid input'
            end

            self
          end

          # Encodes a Rex::Proto::Kerberos::Model::PreAuthDataEntry into an ASN.1 String
          #
          # @return [String]
          def encode
            type_asn1 = OpenSSL::ASN1::ASN1Data.new([encode_type], 1, :CONTEXT_SPECIFIC)
            value_asn1 = OpenSSL::ASN1::ASN1Data.new([encode_value], 2, :CONTEXT_SPECIFIC)
            seq = OpenSSL::ASN1::Sequence.new([type_asn1, value_asn1])

            seq.to_der
          end

          # Gets the value of this PreAuthDataEntry as its instantiated object, based
          # on the type
          #
          # @return [Object] e.g. PreAuthEncTimeStamp, PreAuthEtypeInfo2
          def decoded_value
            case self.type
            when Rex::Proto::Kerberos::Model::PreAuthType::PA_TGS_REQ
              decoded = OpenSSL::ASN1.decode(self.value)
              ApReq.decode(decoded)
            when Rex::Proto::Kerberos::Model::PreAuthType::PA_ENC_TIMESTAMP
              decoded = OpenSSL::ASN1.decode(self.value)
              PreAuthEncTimeStamp.decode(decoded)
            when Rex::Proto::Kerberos::Model::PreAuthType::PA_PW_SALT
              # This is not DER_encoded - just pass the string directly
              PreAuthPwSalt.decode(self.value)
            when Rex::Proto::Kerberos::Model::PreAuthType::PA_ETYPE_INFO
              # Not yet supported
            when Rex::Proto::Kerberos::Model::PreAuthType::PA_PK_AS_REQ
               PreAuthPkAsReq.decode(self.value)
            when Rex::Proto::Kerberos::Model::PreAuthType::PA_PK_AS_REP
               PreAuthPkAsRep.decode(self.value)
            when Rex::Proto::Kerberos::Model::PreAuthType::PA_ETYPE_INFO2
              decoded = OpenSSL::ASN1.decode(self.value)
              PreAuthEtypeInfo2.decode(decoded)
            when Rex::Proto::Kerberos::Model::PreAuthType::PA_PAC_REQUEST
              decoded = OpenSSL::ASN1.decode(self.value)
              PreAuthPacRequest.decode(decoded)
            when Rex::Proto::Kerberos::Model::PreAuthType::PA_FOR_USER
              decoded = OpenSSL::ASN1.decode(self.value)
              PreAuthForUser.decode(decoded)
            else
              # Unknown type - just ignore for now
            end
          end

          private

          # Encodes the type
          #
          # @return [OpenSSL::ASN1::Integer]
          def encode_type
            int_bn = OpenSSL::BN.new(type.to_s)
            int = OpenSSL::ASN1::Integer.new(int_bn)

            int
          end

          # Encodes the value
          #
          # @return [OpenSSL::ASN1::OctetString]
          def encode_value
            OpenSSL::ASN1::OctetString.new(value)
          end

          # Decodes a Rex::Proto::Kerberos::Model::PreAuthDataEntry
          #
          # @param input [String] the input to decode from
          def decode_string(input)
            asn1 = OpenSSL::ASN1.decode(input)

            decode_asn1(asn1)
          end

          # Decodes a Rex::Proto::Kerberos::Model::PreAuthDataEntry from an
          # OpenSSL::ASN1::Sequence
          #
          # @param input [OpenSSL::ASN1::Sequence] the input to decode from
          def decode_asn1(input)
            seq_values = input.value
            self.type  = decode_asn1_type(seq_values[0])
            self.value = decode_asn1_value(seq_values[1])
          end

          # Decodes the type from an OpenSSL::ASN1::ASN1Data
          #
          # @param input [OpenSSL::ASN1::ASN1Data] the input to decode from
          # @return [Integer]
          def decode_asn1_type(input)
            input.value[0].value.to_i
          end

          # Decodes the value from an OpenSSL::ASN1::ASN1Data
          #
          # @param input [OpenSSL::ASN1::ASN1Data] the input to decode from
          # @return [Integer]
          def decode_asn1_value(input)
            input.value[0].value
          end
        end
      end
    end
  end
end