summaryrefslogtreecommitdiff
path: root/spec/support/helpers/key_generator_helper.rb
blob: b1c289ffef7be1215a699344fc7775f3bce12aa7 (plain)
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
module Spec
  module Support
    module Helpers
      class KeyGeneratorHelper
        # The components in a openssh .pub / known_host RSA public key.
        RSA_COMPONENTS = ['ssh-rsa', :e, :n].freeze

        attr_reader :size

        def initialize(size = 2048)
          @size = size
        end

        def generate
          key = OpenSSL::PKey::RSA.generate(size)
          components = RSA_COMPONENTS.map do |component|
            key.respond_to?(component) ? encode_mpi(key.public_send(component)) : component
          end

          # Ruby tries to be helpful and adds new lines every 60 bytes :(
          'ssh-rsa ' + [pack_pubkey_components(components)].pack('m').delete("\n")
        end

        private

        # Encodes an openssh-mpi-encoded integer.
        def encode_mpi(n)
          chars, n = [], n.to_i
          chars << (n & 0xff) && n >>= 8 while n != 0
          chars << 0 if chars.empty? || chars.last >= 0x80
          chars.reverse.pack('C*')
        end

        # Packs string components into an openssh-encoded pubkey.
        def pack_pubkey_components(strings)
          (strings.map { |s| [s.length].pack('N') }).zip(strings).flatten.join
        end
      end
    end
  end
end