2012-01-12 20:20:46 +00:00
##
2014-10-17 16:47:33 +00:00
# This module requires Metasploit: http://metasploit.com/download
2013-10-15 18:50:46 +00:00
# Current source: https://github.com/rapid7/metasploit-framework
2012-01-12 20:20:46 +00:00
##
2016-03-08 13:02:44 +00:00
class MetasploitModule < Msf :: Auxiliary
2013-08-30 21:28:54 +00:00
include Msf :: Exploit :: Remote :: Postgres
include Msf :: Auxiliary :: Report
include Msf :: Auxiliary :: Scanner
2012-01-12 20:20:46 +00:00
2013-08-30 21:28:54 +00:00
def initialize
super (
'Name' = > 'Postgres Schema Dump' ,
'Description' = > %Q{
This module extracts the schema information from a
Postgres server .
} ,
'Author' = > [ 'theLightCosine' ] ,
'License' = > MSF_LICENSE
)
register_options ( [
OptString . new ( 'DATABASE' , [ true , 'The database to authenticate against' , 'postgres' ] ) ,
OptBool . new ( 'DISPLAY_RESULTS' , [ true , " Display the Results to the Screen " , true ] )
] )
deregister_options ( 'SQL' , 'RETURN_ROWSET' , 'VERBOSE' )
2012-01-12 20:20:46 +00:00
2013-08-30 21:28:54 +00:00
end
2012-01-12 20:20:46 +00:00
2013-08-30 21:28:54 +00:00
def run_host ( ip )
pg_schema = get_schema
pg_schema . each do | db |
report_note (
:host = > datastore [ 'RHOST' ] ,
:type = > " postgres.db.schema " ,
:data = > db ,
:port = > datastore [ 'RPORT' ] ,
:proto = > 'tcp' ,
:update = > :unique_data
)
end
output = " Postgres SQL Server Schema \n Host: #{ datastore [ 'RHOST' ] } \n Port: #{ datastore [ 'RPORT' ] } \n ==================== \n \n "
output << YAML . dump ( pg_schema )
this_service = report_service (
:host = > datastore [ 'RHOST' ] ,
:port = > datastore [ 'RPORT' ] ,
:name = > 'postgres' ,
:proto = > 'tcp'
)
store_loot ( 'postgres_schema' , " text/plain " , datastore [ 'RHOST' ] , output , " #{ datastore [ 'RHOST' ] } _postgres_schema.txt " , " Postgres SQL Schema " , this_service )
print_good output if datastore [ 'DISPLAY_RESULTS' ]
end
2012-01-12 20:20:46 +00:00
2013-08-30 21:28:54 +00:00
def get_schema
pg_schema = [ ]
res = smart_query ( 'SELECT datname FROM pg_database' )
if res and not res . empty?
res . each do | row |
defaults = [ 'template1' , 'template0' , 'postgres' ]
next if defaults . include? row [ 0 ]
tmp_db = { }
tmp_db [ 'DBName' ] = row [ 0 ]
tmp_db [ 'Tables' ] = [ ]
postgres_login ( { :database = > row [ 0 ] } )
tmp_tblnames = smart_query ( " SELECT c.relname, n.nspname FROM pg_catalog.pg_class c LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace WHERE n.nspname NOT IN ('pg_catalog','pg_toast') AND pg_catalog.pg_table_is_visible(c.oid); " )
if tmp_tblnames and not tmp_tblnames . empty?
tmp_tblnames . each do | tbl_row |
tmp_tbl = { }
tmp_tbl [ 'TableName' ] = tbl_row [ 0 ]
tmp_tbl [ 'Columns' ] = [ ]
tmp_column_names = smart_query ( " SELECT A.attname, T.typname, A.attlen FROM pg_class C, pg_namespace N, pg_attribute A, pg_type T WHERE (N.oid=C.relnamespace) AND (A.attrelid=C.oid) AND (A.atttypid=T.oid) AND (A.attnum>0) AND (NOT A.attisdropped) AND (N.nspname ILIKE 'public') AND (c.relname=' #{ tbl_row [ 0 ] } '); " )
if tmp_column_names and not tmp_column_names . empty?
tmp_column_names . each do | column_row |
tmp_column = { }
tmp_column [ 'ColumnName' ] = column_row [ 0 ]
tmp_column [ 'ColumnType' ] = column_row [ 1 ]
tmp_column [ 'ColumnLength' ] = column_row [ 2 ]
tmp_tbl [ 'Columns' ] << tmp_column
end
end
tmp_db [ 'Tables' ] << tmp_tbl
end
end
pg_schema << tmp_db
end
end
return pg_schema
end
2012-01-12 20:20:46 +00:00
2013-08-30 21:28:54 +00:00
def smart_query ( query_string )
res = postgres_query ( query_string , false )
2015-04-03 11:12:23 +00:00
# Error handling routine here, borrowed heavily from todb
2013-08-30 21:28:54 +00:00
case res . keys [ 0 ]
when :conn_error
print_error ( " A Connection Error occured " )
return
when :sql_error
case res [ :sql_error ]
when / ^C42501 /
print_error " #{ datastore [ 'RHOST' ] } : #{ datastore [ 'RPORT' ] } Postgres - Insufficent permissions. "
return nil
else
print_error " #{ datastore [ 'RHOST' ] } : #{ datastore [ 'RPORT' ] } Postgres - #{ res [ :sql_error ] } "
return nil
end
when :complete
return res [ :complete ] . rows
end
end
2012-01-12 20:20:46 +00:00
end