#
# Copyright (c) 2006-2012 Hal Brodigan (postmodern.mod3 at gmail.com)
#
# This file is part of Ronin Support.
#
# Ronin Support is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published
# by the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# Ronin Support is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with Ronin Support. If not, see .
#
require 'ronin/extensions/regexp'
require 'ronin/fuzzing/fuzzing'
require 'combinatorics/generator'
require 'combinatorics/list_comprehension'
require 'combinatorics/power_set'
require 'chars'
class String
#
# Generate permutations of Strings from a format template.
#
# @param [Array(, )] template
# The template which defines the string or character sets which will
# make up parts of the String.
#
# @yield [string]
# The given block will be passed each unique String.
#
# @yieldparam [String] string
# A newly generated String.
#
# @return [Enumerator]
# If no block is given, an Enumerator will be returned.
#
# @raise [ArgumentError]
# A given character set name was unknown.
#
# @raise [TypeError]
# A given string set was not a String, Symbol or Enumerable.
# A given string set length was not an Integer or Enumerable.
#
# @example Generate Strings with ranges of repeating sub-strings.
#
# @example Generate Strings with three alpha chars and one numeric chars.
# String.generate([:alpha, 3], :numeric) do |password|
# puts password
# end
#
# @example Generate Strings with two to four alpha chars.
# String.generate([:alpha, 2..4]) do |password|
# puts password
# end
#
# @example Generate Strings using alpha and punctuation chars.
# String.generate([Chars.alpha + Chars.punctuation, 4]) do |password|
# puts password
# end
#
# @example Generate Strings from a custom char set.
# String.generate([['a', 'b', 'c'], 3], [['1', '2', '3'], 3]) do |password|
# puts password
# end
#
# @example Generate Strings containing known Strings.
# String.generate("rock", [:numeric, 4]) do |password|
# puts password
# end
#
# @example Generate Strings with ranges of repeating sub-strings.
# String.generate(['/AA', (1..100).step(5)]) do |path|
# puts path
# end
#
# @since 0.3.0
#
# @api public
#
def self.generate(*template)
return enum_for(:generate,*template) unless block_given?
sets = []
template.each do |pattern|
set, length = pattern
set = case set
when String
[set].each
when Symbol
name = set.to_s.upcase
unless Chars.const_defined?(name)
raise(ArgumentError,"unknown charset #{set.inspect}")
end
Chars.const_get(name).each_char
when Enumerable
set
else
raise(TypeError,"set must be a String, Symbol or Enumerable")
end
case length
when Integer
length.times { sets << set.dup }
when Array, Range
sets << Combinatorics::Generator.new do |g|
length.each do |sublength|
superset = Array.new(sublength) { set.dup }
superset.comprehension { |strings| g.yield strings.join }
end
end
when nil
sets << set
else
raise(TypeError,"length must be an Integer, Range or Array")
end
end
sets.comprehension do |strings|
new_string = ''
strings.each do |string|
new_string << case string
when Integer
string.chr
else
string.to_s
end
end
yield new_string
end
return nil
end
#
# Repeats the String.
#
# @param [Enumerable, Integer] n
# The number of times to repeat the String.
#
# @yield [repeated]
# The given block will be passed every repeated String.
#
# @yieldparam [String] repeated
# A repeated version of the String.
#
# @return [Enumerator]
# If no block is given, an Enumerator will be returned.
#
# @raise [TypeError]
# `n` must either be Enumerable or an Integer.
#
# @example
# 'A'.repeating(100)
# # => "AAAAAAAAAAAAA..."
#
# @example Generates 100 upto 700 `A`s, increasing by 100 at a time:
# 'A'.repeating((100..700).step(100)) do |str|
# # ...
# end
#
# @example Generates 128, 1024, 65536 `A`s:
# 'A'.repeating([128, 1024, 65536]) do |str|
# # ...
# end
#
# @api public
#
# @since 0.4.0
#
def repeating(n)
if n.kind_of?(Integer)
# if n is an Integer, simply multiply the String and return
repeated = (self * n)
yield repeated if block_given?
return repeated
end
return enum_for(:repeating,n) unless block_given?
unless n.kind_of?(Enumerable)
raise(TypeError,"argument must be Enumerable or an Integer")
end
n.each do |length|
yield(self * length)
end
return self
end
#
# Incrementally fuzzes the String.
#
# @param [Hash{Regexp,String => #each}] substitutions
# Patterns and their substitutions.
#
# @yield [fuzz]
# The given block will be passed every fuzzed String.
#
# @yieldparam [String] fuzz
# A fuzzed String.
#
# @return [Enumerator]
# If no block is given, an Enumerator will be returned.
#
# @example Replace every `e`, `i`, `o`, `u` with `(`, 100 `A`s and a `\0`:
# "the quick brown fox".fuzz(/[eiou]/ => ['(', ('A' * 100), "\0"]) do |str|
# p str
# end
#
# @example {String.generate} with {String#fuzz}:
# "GET /".fuzz('/' => String.generate(['A', 1..100])) do |str|
# p str
# end
#
# @since 0.3.0
#
# @api public
#
def fuzz(substitutions={})
return enum_for(:fuzz,substitutions) unless block_given?
substitutions.each do |pattern,substitution|
pattern = case pattern
when Regexp
pattern
when String
Regexp.new(Regexp.escape(pattern))
when Symbol
Regexp.const_get(pattern.to_s.upcase)
else
raise(TypeError,"cannot convert #{pattern.inspect} to a Regexp")
end
substitution = case substitution
when Enumerable
substitution
when Symbol
Ronin::Fuzzing[substitution]
else
raise(TypeError,"substitutions must be Enumerable or a Symbol")
end
scanner = StringScanner.new(self)
indices = []
while scanner.scan_until(pattern)
indices << [scanner.pos - scanner.matched_size, scanner.matched_size]
end
indices.each do |index,length|
substitution.each do |substitute|
substitute = case substitute
when Proc
substitute.call(self[index,length])
when Integer
substitute.chr
else
substitute.to_s
end
fuzzed = dup
fuzzed[index,length] = substitute
yield fuzzed
end
end
end
end
#
# Permutes over every possible mutation of the String.
#
# @param [Hash{Regexp,String,Symbol => Symbol,#each}] mutations
# The patterns and substitutions to mutate the String with.
#
# @yield [mutant]
# The given block will be yielded every possible mutant String.
#
# @yieldparam [String] mutant
# A mutated String.
#
# @return [Enumerator]
# If no block is given, an Enumerator will be returned.
#
# @example
# "hello old dog".mutate('e' => ['3'], 'l' => ['1'], 'o' => ['0']) do |str|
# puts str
# end
#
# @since 0.4.0
#
# @api public
#
def mutate(mutations={})
return enum_for(:mutate,mutations) unless block_given?
matches = Set[]
mutations.each do |pattern,mutation|
pattern = case pattern
when Regexp
pattern
when String
Regexp.new(Regexp.escape(pattern))
when Symbol
Regexp.const_get(pattern.to_s.upcase)
else
raise(TypeError,"cannot convert #{pattern.inspect} to a Regexp")
end
scanner = StringScanner.new(self)
while scanner.scan_until(pattern)
length = scanner.matched_size
index = scanner.pos - length
original = scanner.matched
mutator = Combinatorics::Generator.new do |g|
mutation.each do |mutate|
g.yield case mutate
when Proc
mutate.call(original)
when Integer
mutate.chr
else
mutate.to_s
end
end
end
matches << [index, length, mutator]
end
end
matches.powerset do |submatches|
# ignore the empty Set
next if submatches.empty?
# sort the submatches by index
submatches = submatches.sort_by { |index,length,mutator| index }
sets = []
prev_index = 0
submatches.each do |index,length,mutator|
# add the previous substring to the set of Strings
if index > prev_index
sets << [self[prev_index,index - prev_index]]
end
# add the mutator to the set of Strings
sets << mutator
prev_index = index + length
end
# add the remaining substring to the set of Strings
if prev_index < self.length
sets << [self[prev_index..-1]]
end
sets.comprehension { |strings| yield strings.join }
end
end
end