module Msf module Ui module Gtk2 ### # # This class help us to wait the next release of ruby-libglade2 package # ### class GladeXML < GladeXML def connect(source, target, signal, handler, data, after = false) @handler_proc ||= Proc.new{} handler = canonical_handler(handler) if target signal_proc = target.method(handler) else signal_proc = @handler_proc.call(handler) end if after sig_conn_proc = source.method(:signal_connect_after) else sig_conn_proc = source.method(:signal_connect) end if signal_proc guard_source_from_gc(source) case signal_proc.arity when 0 sig_conn_proc.call(signal) {signal_proc.call} else sig_conn_proc.call(signal, &signal_proc) end end end def guard_source_from_gc(source) return if source.nil? @sources ||= {} @sources[source.object_id] = source source.signal_connect("destroy") do |object| @sources.delete(object.object_id) end # To get the parent window of the source as a ruby object. # Ruby/GTK keeps the Window objects on the memory to prevend from GC. parent = source.parent while parent parent = parent.parent end end end ### # # This class help us to retreive all glade widgets and place them in your user instance # like @windows, @widget, ... # ### class MyGlade include Msf::Ui::Gtk2::MyControls def initialize(root) # Give the glade file and instance the glade object file_glade = ::Rex::Compat.temp_copy(File.join(driver.resource_directory, 'msfgui.glade')) glade = GladeXML.new(file_glade.path, root) { |handler| method(handler) } # For all widget names, instance a variable glade.widget_names.each do |name| begin instance_variable_set("@#{name}".intern, glade[name]) rescue end end ::Rex::Compat.temp_delete(file_glade) end end ### # # This class provides the splash screen # ### class MySplash < Gtk::Window include Msf::Ui::Gtk2::MyControls def initialize super(Gtk::Window::TOPLEVEL) set_type_hint(Gdk::Window::TYPE_HINT_SPLASHSCREEN) set_decorated(false) set_window_position(Gtk::Window::POS_CENTER) set_default_size(460,301) set_border_width(5) image = Gtk::Image.new(driver.get_icon("splash.xpm")) add(image) show_all end end ### # # This class provides the main window. # ### class MyApp < MyGlade attr_accessor :window include Msf::Ui::Gtk2::MyControls def initialize # console_style = File.join(driver.resource_directory, 'style', 'main.rc') # Gtk::RC.parse(console_style) super('window') # Set a default icon for all widgets Gtk::Window.set_default_icon(driver.get_icon('msfwx.xpm')) @window.set_icon(driver.get_icon('msfwx.xpm')) # Set a title with the version @window.set_title("Metasploit Framework GUI v#{Msf::Framework::Version}") # Destroy @window.signal_connect('destroy') { Gtk.main_quit } # Default size # @window.set_default_size(1024, 768) # Defaults policies for Gtk::ScrolledWindow @scrolledwindow_module.set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC) @scrolledwindow_jobs.set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC) @scrolledwindow_session.set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC) @scrolledwindow_information.set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC) @scrolledwindow_logs.set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC) # Logs Buffer @buffer = Gtk::TextBuffer.new @viewlogs.set_buffer(@buffer_logs) @viewlogs.set_editable(false) @viewlogs.set_cursor_visible(false) # Sessions Tree @session_tree = MySessionTree.new(@treeview_session) # Target Tree @job_tree = MyJobTree.new(@treeview2) # Module Tree @module_tree = MyModuleTree.new(@treeview1, @viewmodule) # Tooltips tooltips = Gtk::Tooltips.new # Configure the window handles for easy reference $gtk2driver.main = @window $gtk2driver.session_tree = @session_tree $gtk2driver.job_tree = @job_tree $gtk2driver.module_tree = @module_tree $gtk2driver.log_text = @viewlogs $gtk2driver.tips = tooltips # Initialize the search class ModuleSearch.new(@search_entry, @search_button, @search_cancel_button) # Focus on the search widget @search_entry.can_focus = true @search_entry.grab_focus # Update the StatusBar with all framework modules refresh() # Print a welcome message $gtk2driver.append_log_view("Initialized the Metasploit Framework GUI.\n") # Handle any initalization options process_cli_options() end def process_cli_options res = $gtk2driver.opts['Resource'] || return data = "" begin data = File.read(res) rescue ::Exception => e $gtk2driver.append_log_view("Failed to open the resource file: #{e}\n") return end # spawn a new console pipe = MsfWindow::Consoles.new.last_console.pipe data.each_line do |line| line.strip! next if line.length == 0 next if line =~ /^#/ $gtk2driver.append_log_view("Executing: #{line}...\n") pipe.write_input(line + "\n") end end # # Actions for options stuff (databases, global variable, ...) # def on_options_activate MsfPreferences::Main.new() end # # Signal to refresh the treeview module # def on_refresh_activate refresh() end # # Bye bye # def on_leave_activate Gtk.main_quit end # # Actions for OpCodes/Stats # def on_stats_activate MsfOpcode::Stats.new() end # # Actions for OpCodes/Locales # def on_locales_activate MsfOpcode::Locales.new() end # # Actions for OpCodes/Metatypes # def on_metatypes_activate MsfOpcode::Metatypes.new() end # # Actions for OpCodes/Groups # def on_groups_activate MsfOpcode::Groups.new() end # # Actions for OpCodes/Types # def on_types_activate MsfOpcode::Types.new() end # # Actions for OpCodes/Platforms # def on_platforms_activate MsfOpcode::Platforms.new() end # # Actions for OpCodes/Modules # def on_modules_activate MsfOpcode::Modules.new() end # # Actions for OpCodes/Search # def on_search_activate end # # Action for "Window/Logs" menu # def on_logs_activate MsfWindow::Logs.new end # # Action for "Window/Console" menu # def on_consoles_activate MsfWindow::Consoles.new end # # The About Dialog # def on_about_activate ad = MyAbout.new(@window) ad.signal_connect('response'){ ad.destroy } ad.show end # # The Online Help link # def on_online_activate Rex::Compat.open_browser('http://metasploit.com/framework/support') end # # Call the refresh method to reload all module # def refresh @module_tree.refresh context_id = @statusbar.get_context_id("update") @statusbar.push( context_id, "Loaded " + framework.stats.num_exploits.to_s + " exploits, " + framework.stats.num_payloads.to_s + " payloads, " + framework.stats.num_encoders.to_s + " encoders, " + framework.stats.num_nops.to_s + " nops, and " + framework.stats.num_auxiliary.to_s + " auxiliary" ) end end end end end