# encoding: utf-8

module RubyProf
  # Helper class to simplify printing profiles of several types from
  # one profiling run. Currently prints a flat profile, a callgrind
  # profile, a call stack profile and a graph profile.
  class MultiPrinter
    def initialize(result, printers = [:flat, :graph_html])
      @flat_printer = FlatPrinter.new(result) if printers.include?(:flat)

      @graph_printer = GraphPrinter.new(result) if printers.include?(:graph)
      @graph_html_printer = GraphHtmlPrinter.new(result) if printers.include?(:graph_html)

      @tree_printer = CallTreePrinter.new(result) if printers.include?(:tree)
      @call_info_printer = CallInfoPrinter.new(result) if printers.include?(:call_tree)

      @stack_printer = CallStackPrinter.new(result) if printers.include?(:stack)
      @dot_printer = DotPrinter.new(result) if printers.include?(:dot)
    end

    def self.needs_dir?
      true
    end

    # create profile files under options[:path] or the current
    # directory. options[:profile] is used as the base name for the
    # profile file, defaults to "profile".
    def print(options)
      validate_print_params(options)

      @profile = options.delete(:profile) || "profile"
      @directory = options.delete(:path) || File.expand_path(".")

      print_to_flat(options) if @flat_printer

      print_to_graph(options) if @graph_printer
      print_to_graph_html(options) if @graph_html_printer

      print_to_stack(options) if @stack_printer
      print_to_call_info(options) if @call_info_printer
      print_to_tree(options) if @tree_printer
      print_to_dot(options) if @dot_printer
    end

    # the name of the flat profile file
    def flat_report
      "#{@directory}/#{@profile}.flat.txt"
    end

    # the name of the graph profile file
    def graph_report
      "#{@directory}/#{@profile}.graph.txt"
    end

    def graph_html_report
      "#{@directory}/#{@profile}.graph.html"
    end

    # the name of the callinfo profile file
    def call_info_report
      "#{@directory}/#{@profile}.call_tree.txt"
    end

    # the name of the callgrind profile file
    def tree_report
      "#{@directory}/#{@profile}.callgrind.out.#{$$}"
    end

    # the name of the call stack profile file
    def stack_report
      "#{@directory}/#{@profile}.stack.html"
    end

    # the name of the call stack profile file
    def dot_report
      "#{@directory}/#{@profile}.dot"
    end

    def print_to_flat(options)
      File.open(flat_report, "wb") do |file|
        @flat_printer.print(file, options)
      end
    end

    def print_to_graph(options)
      File.open(graph_report, "wb") do |file|
        @graph_printer.print(file, options)
      end
    end

    def print_to_graph_html(options)
      File.open(graph_html_report, "wb") do |file|
        @graph_html_printer.print(file, options)
      end
    end

    def print_to_call_info(options)
      File.open(call_info_report, "wb") do |file|
        @call_info_printer.print(file, options)
      end
    end

    def print_to_tree(options)
      @tree_printer.print(options.merge(:path => @directory, :profile => @profile))
    end

    def print_to_stack(options)
      File.open(stack_report, "wb") do |file|
        @stack_printer.print(file, options.merge(:graph => "#{@profile}.graph.html"))
      end
    end

    def print_to_dot(options)
      File.open(dot_report, "wb") do |file|
        @dot_printer.print(file, options)
      end
    end

    def validate_print_params(options)
      if options.is_a?(IO)
        raise ArgumentError, "#{self.class.name}#print cannot print to IO objects"
      elsif !options.is_a?(Hash)
        raise ArgumentError, "#{self.class.name}#print requires an options hash"
      end
    end
  end
end
