lib/base32.rb in base32-0.2.0 vs lib/base32.rb in base32-0.3.0

- old
+ new

@@ -1,26 +1,29 @@ +require 'openssl' + +# Module for encoding and decoding in Base32 per RFC 3548 module Base32 - TABLE = "ABCDEFGHIJKLMNOPQRSTUVWXYZ234567" + TABLE = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ234567'.freeze class Chunk def initialize(bytes) @bytes = bytes end def decode bytes = @bytes.take_while {|c| c != 61} # strip padding n = (bytes.length * 5.0 / 8.0).floor p = bytes.length < 8 ? 5 - (n * 8) % 5 : 0 - c = bytes.inject(0) {|m,o| (m << 5) + TABLE.index(o.chr)} >> p + c = bytes.inject(0) {|m,o| (m << 5) + Base32.table.index(o.chr)} >> p (0..n-1).to_a.reverse.collect {|i| ((c >> i * 8) & 0xff).chr} end - + def encode n = (@bytes.length * 8.0 / 5.0).ceil p = n < 8 ? 5 - (@bytes.length * 8) % 5 : 0 c = @bytes.inject(0) {|m,o| (m << 8) + o} << p - [(0..n-1).to_a.reverse.collect {|i| TABLE[(c >> i * 5) & 0x1f].chr}, + [(0..n-1).to_a.reverse.collect {|i| Base32.table[(c >> i * 5) & 0x1f].chr}, ("=" * (8-n))] end end def self.chunks(str, size) @@ -30,14 +33,35 @@ result << Chunk.new(bytes.take(size)) bytes = bytes.drop(size) end result end - + def self.encode(str) chunks(str, 5).collect(&:encode).flatten.join end def self.decode(str) chunks(str, 8).collect(&:decode).flatten.join + end + + def self.random_base32(length=16) + random = '' + OpenSSL::Random.random_bytes(length).each_byte do |b| + random << self.table[b % 32] + end + random.ljust((length / 8.0).ceil * 8, '=') # add padding + end + + def self.table=(table) + raise ArgumentError, "Table must have 32 unique characters" unless self.table_valid?(table) + @table = table + end + + def self.table + @table || TABLE + end + + def self.table_valid?(table) + table.bytes.size == 32 && table.bytes.uniq.size == 32 end end