module DummyLogGenerator class Generator def initialize(setting) prepare_format_proc(setting.labeled, setting.delimiter) prepare_field_procs(setting.fields) end def prepare_format_proc(labeled, delimiter) @format_proc = if labeled Proc.new {|fields| fields.map {|key, val| "#{key}:#{val}" }.join(delimiter) } else Proc.new {|fields| fields.values.join(delimiter) } end end def prepare_field_procs(fields) rand = ::DummyLogGenerator::Random.new @field_procs = {} fields.each do |key, opts| opts = opts.dup type = opts.delete(:type) if rand.respond_to?(type) @field_procs[key] = rand.send(type, opts) else raise ConfigError.new(type) end end end def generate(prev_data = {}) data = {} @field_procs.each do |key, proc| prev = prev_data[key] || -1 data[key] = proc.call(prev) end data end def format(fields) @format_proc.call(fields) end end class Random def initialize @rand = ::Random.new(0) @chars = ('a'..'z').to_a + ('A'..'Z').to_a + ('0'..'9').to_a # no symbols and multi-bytes for now end def string(length: 8, any: nil, value: nil) if value string = value.to_s Proc.new { string } elsif any Proc.new { self.any(any) } else Proc.new { Array.new(length){@chars[rand(@chars.size-1)]}.join } end end def integer(format: nil, range: nil, countup: false, value: nil) if format if value integer = sprintf(format, value.to_i) Proc.new { integer } elsif range Proc.new { sprintf(format, self.range(range)) } elsif countup Proc.new {|prev| sprintf(format, prev.to_i + 1) } else Proc.new { sprintf(format, rand(0..2,147,483,647)) } end else if value integer = value.to_i Proc.new { integer } elsif range Proc.new { self.range(range) } elsif countup Proc.new {|prev| prev + 1 } else Proc.new { rand(0..2,147,483,647) } end end end def float(format: nil, range: nil, value: nil) if format if value float = value.to_f Proc.new { sprintf(format, float) } elsif range Proc.new { sprintf(format, self.range(range)) } else Proc.new { r = rand(1..358); sprintf(format, r * Math.cos(r)) } end else if value float = value.to_f Proc.new { float } elsif range Proc.new { self.range(range) } else Proc.new { r = rand(1..358); r * Math.cos(r) } end end end def datetime(format: "%Y-%m-%d %H:%M:%S.%3N", random: false, value: nil) if value Proc.new { value.sprintf(format) } elsif random Proc.new { y = rand(1970..2037); m = rand(1..12); d = rand(1..27); h = rand(0..23); min = rand(0..59); s = rand(0..59); usec = rand(0..999999); Time.local(y, m, d, h, min, s, usec) } else Proc.new { Time.now } end end def range(range) rand(range) end def any(any) any[rand(any.size-1)] end def rand(arg = nil) @rand.rand(arg) end end end