File: channel_binding.rb

package info (click to toggle)
ruby-ntlm 0.6.3-2
  • links: PTS, VCS
  • area: main
  • in suites: bookworm, forky, sid, trixie
  • size: 408 kB
  • sloc: ruby: 2,663; makefile: 6
file content (65 lines) | stat: -rw-r--r-- 2,256 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
module Net
  module NTLM
    class ChannelBinding

      # Creates a ChannelBinding used for Extended Protection Authentication
      # @see http://blogs.msdn.com/b/openspecification/archive/2013/03/26/ntlm-and-channel-binding-hash-aka-exteneded-protection-for-authentication.aspx
      #
      # @param outer_channel [OpenSSL::X509::Certificate] Server certificate securing
      #   the outer TLS channel
      # @return [NTLM::ChannelBinding] A ChannelBinding holding a token that can be
      #   embedded in a {Type3} message
      def self.create(outer_channel)
        new(outer_channel)
      end

      # @param outer_channel [OpenSSL::X509::Certificate] Server certificate securing
      #   the outer TLS channel
      def initialize(outer_channel)
        @channel = outer_channel
        @unique_prefix = 'tls-server-end-point'
        @initiator_addtype = 0
        @initiator_address_length = 0
        @acceptor_addrtype = 0
        @acceptor_address_length = 0
      end

      attr_reader :channel, :unique_prefix, :initiator_addtype
      attr_reader :initiator_address_length, :acceptor_addrtype
      attr_reader :acceptor_address_length

      # Returns a channel binding hash acceptable for use as a AV_PAIR MsvAvChannelBindings
      #   field value as specified in the NTLM protocol
      #
      # @return [String] MD5 hash of gss_channel_bindings_struct
      def channel_binding_token
        @channel_binding_token ||= OpenSSL::Digest::MD5.new(gss_channel_bindings_struct).digest
      end

      def gss_channel_bindings_struct
        @gss_channel_bindings_struct ||= begin
          token = [initiator_addtype].pack('I')
          token << [initiator_address_length].pack('I')
          token << [acceptor_addrtype].pack('I')
          token << [acceptor_address_length].pack('I')
          token << [application_data.length].pack('I')
          token << application_data
          token
        end
      end

      def channel_hash
        @channel_hash ||= OpenSSL::Digest::SHA256.new(channel.to_der)
      end

      def application_data
        @application_data ||= begin
          data = unique_prefix
          data << ':'
          data << channel_hash.digest
          data
        end
      end
    end
  end
end