-
Notifications
You must be signed in to change notification settings - Fork 346
/
Search.py
406 lines (391 loc) · 17.7 KB
/
Search.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
#!/usr/bin/python
# -*- coding: utf-8 -*-
from OracleDatabase import OracleDatabase
import logging
from Constants import *
from Utils import checkOptionsGivenByTheUser, getScreenSize
from texttable import Texttable
class Search (OracleDatabase):
'''
Serach in Databases, tables and columns
'''
REQ_INFO_FROM_COLUMN_NAMES = "SELECT owner, table_name, column_name FROM all_tab_columns WHERE column_name LIKE '{0}'" #{0}==pattern
REQ_VALUE_IN_COLUMN = 'SELECT "{0}" FROM "{1}"."{2}" WHERE "{0}" is not null and rownum = 1' #{0}==column, {1}==database, {2}==table
REQ_GET_ALL_NO_SYSTEM_TABLES = "SELECT DISTINCT owner, table_name FROM all_tables WHERE owner not in ('SYS','SYSTEM')"
REQ_GET_COLUMNS_FOR_TABLE = "SELECT column_name, data_type FROM all_tab_columns WHERE table_name='{0}' and owner='{1}'" #{0}==table name, {1}==owner
DEFAULT_VALUE_EMPTY_COLUMN = "(Empty Column)"
DEFAULT_VALUE_UNKNOWN = "(Unknown)"
EXEMPLE_VALUE_LEN_MAX = 40
TRUNCATED_MESSAGE_EXEMPLE = '(Truncated...)'
def __init__(self,args):
'''
Constructor
'''
logging.debug("Search object created")
OracleDatabase.__init__(self,args)
def searchInColumns(self, sqlPattern, showEmptyColumns, withoutExample=False):
'''
Search sqlpattern in all columns names
returns a list which contains dicos ex: [{'columnName': 'passwd', 'tableName':'users', 'database':'mysite'}{'columnName': 'password', 'tableName':'users', 'database':'mysitetemp'}]
'''
logging.info("Searching pattern '{0}' in column names".format(sqlPattern.upper()))
results = self.__execQuery__(query=self.REQ_INFO_FROM_COLUMN_NAMES.format(sqlPattern.upper()), ld=['owner', 'table_name', 'column_name'])
if isinstance(results, Exception):
logging.error("Impossible to continue in searchInColumns(): {0}".format(results))
return []
table = self.getInfoIntable(results, ["owner","table_name","column_name", "example"], showEmptyColumns=showEmptyColumns, withoutExample=withoutExample)
return table
def searchPwdKeyworkInColumnNames(self, showEmptyColumns, withoutExample=False):
'''
Search sqlpattern in all columns names
returns a list which contains dicos ex: [{'columnName': 'passwd', 'tableName':'users', 'database':'mysite'}{'columnName': 'password', 'tableName':'users', 'database':'mysitetemp'}]
'''
tables = ""
for aPattern in PATTERNS_COLUMNS_WITH_PWDS:
table = self.searchInColumns(aPattern, showEmptyColumns=showEmptyColumns, withoutExample=withoutExample)
if self.isEmptyTable(table) == True :
logging.debug("Nothing to print. Doesn't print the header of the table!")
else :
logging.debug("Some results, saved")
try :
tables += "'"+aPattern+"' in column names:\n"+table+"\n\n"
except UnicodeDecodeError as e:
print("------->"+e)
return tables
def getDescOfEachNoSystemTable(self):
'''
returns a String for print
'''
outputString = ""
logging.debug("Getting all no system tables accessible with the current user")
tablesAccessible = self.__execQuery__(query=self.REQ_GET_ALL_NO_SYSTEM_TABLES, ld=['owner', 'table_name'])
if isinstance(tablesAccessible,Exception):
logging.warning("Impossible to execute the request '{0}': {1}".format(self.REQ_GET_ALL_NO_SYSTEM_TABLES, tablesAccessible.generateInfoAboutError(self.REQ_GET_ALL_NO_SYSTEM_TABLES)))
return ""
else:
nbTables = len(tablesAccessible)
colNb = nbTables
if colNb>0 :
pbar,currentColNum = self.getStandardBarStarted(colNb), 0
for aTable in tablesAccessible:
if colNb>0:
currentColNum += 1
pbar.update(currentColNum)
request = self.REQ_GET_COLUMNS_FOR_TABLE.format(aTable['table_name'], aTable['owner'])
columnsAndTypes = self.__execQuery__(query=request, ld=['column_name', 'data_type'])
if isinstance(columnsAndTypes,Exception):
logging.warning("Impossible to execute the request '{0}': {1}".format(request, columnsAndTypes.generateInfoAboutError(request)))
outputString += "\n[+] {0}.{1} ({2}/{3})\n".format(aTable['owner'], aTable['table_name'], currentColNum, colNb)
resultsToTable = [('column_name', 'data_type')]
for aLine in columnsAndTypes:
resultsToTable.append((aLine['column_name'], aLine['data_type']))
table = Texttable(max_width=getScreenSize()[0])
table.set_deco(Texttable.HEADER)
table.add_rows(resultsToTable)
outputString += table.draw()
outputString += '\n'
if colNb>0 : pbar.finish()
return outputString
def getInfoIntable(self,listOfDicos, columns, showEmptyColumns, withoutExample=False):
'''
columns: list which contains column names for the output
returns a String for print
'''
isStringValueInColumn = False
resultsToTable = [columns]
colNb = len(listOfDicos)
if colNb>0 : pbar,currentColNum = self.getStandardBarStarted(colNb), 0
for e in listOfDicos :
isStringValueInColumn = False
if colNb>0 : currentColNum += 1
if colNb>0 : pbar.update(currentColNum)
l = []
l.append(e['owner'])
l.append(e['table_name'])
l.append(e['column_name'])
if withoutExample == True:
l.append(self.DEFAULT_VALUE_UNKNOWN)
resultsToTable.append(l)
else:
logging.debug("Search a not null value in the column '{0}' of the table '{1}'.'{2}' ({3}/{4})".format(e['column_name'], e['owner'], e['table_name'],currentColNum,colNb))
req_value_in_column = self.REQ_VALUE_IN_COLUMN.format(e['column_name'],e['owner'],e['table_name'])
aValue = self.__execQuery__(query=req_value_in_column, ld=['value'])
if isinstance(aValue,Exception):
logging.warning("Impossible to execute the request '{0}' in column names: {1}".format(req_value_in_column, aValue.generateInfoAboutError(req_value_in_column)))
l.append(self.DEFAULT_VALUE_UNKNOWN)
elif aValue == [] : l.append(self.DEFAULT_VALUE_EMPTY_COLUMN)
else :
value = aValue[0]['value']
if type(value) is str:
isStringValueInColumn = True
if len(value) > self.EXEMPLE_VALUE_LEN_MAX:
value = value[0:self.EXEMPLE_VALUE_LEN_MAX] + ' ' +self.TRUNCATED_MESSAGE_EXEMPLE
l.append(value)
else: l.append(self.DEFAULT_VALUE_EMPTY_COLUMN)
if isStringValueInColumn == True :
resultsToTable.append(l)
elif showEmptyColumns==True :
resultsToTable.append(l)
if colNb>0 : pbar.finish()
table = Texttable(max_width=getScreenSize()[0])
table.set_deco(Texttable.HEADER)
table.add_rows(resultsToTable)
return table.draw()
def isEmptyTable (self, table):
"""
String table
"""
if table.count('\n') <= 1 :
return True
else :
return False
def startInteractiveSQLShell(self):
"""
Start an interactive SQL shell
Return True when finished
Tested with:
- select
- create user
- create or replace
"""
print("Ctrl-D to close the SQL shell")
while True:
theLine = None
allLines = ""
print("SQL> ", end='')
while theLine != "":
try:
theLine = input()
except EOFError:
print("\nSQL shell closed")
return True
allLines += theLine
if allLines != "":
results = self.__execQuery__(query=allLines, getColumnNames=True,stringOnly=True)
if isinstance(results,Exception):
print(results)
elif results==[()]:
print("Executed successfully")
else:
table = Texttable(max_width=getScreenSize()[0])
table.set_deco(Texttable.HEADER)
table.add_rows(results)
print(table.draw())
def getAllPrivs(self):
'''
Get all DBA users
Use the method of Alexander Kornbrust
See details here https://www.doag.org/formes/pubfiles/11859287/2019-DB-Alexander_Kornbrust-Best_of_Oracle_Security_2019-Praesentation.pdf
:return:
'''
randomViewName = generateRandomString(length=10)
REQ_CREATE_VIEW = """
create or replace view v_all_privs as
SELECT PRIVILEGE, OBJ_OWNER, OBJ_NAME, USERNAME,COMMON,
LISTAGG(GRANT_TARGET, ',') WITHIN GROUP (ORDER BY GRANT_TARGET) AS GRANT_SOURCES,
MAX(ADMIN_OR_GRANT_OPT) AS ADMIN_OR_GRANT_OPT,
MAX(HIERARCHY_OPT) AS HIERARCHY_OPT
FROM (
WITH ALL_ROLES_FOR_USER AS (SELECT DISTINCT CONNECT_BY_ROOT GRANTEE AS GRANTED_USER, GRANTED_ROLE FROM DBA_ROLE_PRIVS CONNECT BY GRANTEE = PRIOR GRANTED_ROLE)
SELECT PRIVILEGE, OBJ_OWNER, OBJ_NAME, USERNAME, COMMON, REPLACE(GRANT_TARGET, USERNAME, 'Direct to user') AS GRANT_TARGET,ADMIN_OR_GRANT_OPT, HIERARCHY_OPT
FROM (
SELECT distinct PRIVILEGE, NULL AS OBJ_OWNER, NULL AS OBJ_NAME, GRANTEE AS USERNAME, COMMON, GRANTEE AS GRANT_TARGET, ADMIN_OPTION AS ADMIN_OR_GRANT_OPT, NULL AS HIERARCHY_OPT
FROM DBA_SYS_PRIVS
WHERE GRANTEE IN (SELECT USERNAME FROM DBA_USERS)
UNION ALL
SELECT PRIVILEGE, NULL AS OBJ_OWNER, NULL AS OBJ_NAME, ALL_ROLES_FOR_USER.GRANTED_USER AS USERNAME, COMMON, GRANTEE AS GRANT_TARGET, ADMIN_OPTION AS ADMIN_OR_GRANT_OPT, NULL AS HIERARCHY_OPT
FROM DBA_SYS_PRIVS
JOIN ALL_ROLES_FOR_USER ON ALL_ROLES_FOR_USER.GRANTED_ROLE = DBA_SYS_PRIVS.GRANTEE
UNION ALL
SELECT distinct PRIVILEGE, OWNER AS OBJ_OWNER, TABLE_NAME AS OBJ_NAME, GRANTEE AS USERNAME, COMMON, GRANTEE AS GRANT_TARGET, GRANTABLE, HIERARCHY
FROM DBA_TAB_PRIVS
WHERE GRANTEE IN (SELECT USERNAME FROM DBA_USERS)
UNION ALL
SELECT distinct PRIVILEGE, OWNER AS OBJ_OWNER, TABLE_NAME AS OBJ_NAME, GRANTEE AS USERNAME, COMMON, ALL_ROLES_FOR_USER.GRANTED_ROLE AS GRANT_TARGET, GRANTABLE, HIERARCHY
FROM DBA_TAB_PRIVS
JOIN ALL_ROLES_FOR_USER ON ALL_ROLES_FOR_USER.GRANTED_ROLE = DBA_TAB_PRIVS.GRANTEE
union all
SELECT distinct PRIVILEGE, OWNER AS OBJ_OWNER, TABLE_NAME||'.'||COLUMN_NAME AS OBJ_NAME, GRANTEE AS USERNAME, COMMON, ALL_ROLES_FOR_USER.GRANTED_ROLE AS GRANT_TARGET, GRANTABLE, null
FROM DBA_COL_PRIVS
JOIN ALL_ROLES_FOR_USER ON ALL_ROLES_FOR_USER.GRANTED_ROLE = DBA_COL_PRIVS.GRANTEE
UNION ALL
SELECT distinct PRIVILEGE, OWNER AS OBJ_OWNER, TABLE_NAME||'.'||COLUMN_NAME AS OBJ_NAME, GRANTEE AS USERNAME, COMMON, GRANTEE AS GRANT_TARGET, GRANTABLE, null
FROM DBA_COL_PRIVS
WHERE GRANTEE IN (SELECT USERNAME FROM DBA_USERS)
) ALL_USER_PRIVS
) DISTINCT_USER_PRIVS
GROUP BY PRIVILEGE, OBJ_OWNER, OBJ_NAME, USERNAME, COMMON
""".format(randomViewName)
REQ_DELETE_VIEW = "DROP VIEW oracle_ocm.{0}".format(randomViewName)
def getBasicInformation(self, printStdout=True):
'''
Get basic information about the instance and database
:return: Dictionary
'''
logging.info("getting basic information about instance and database...")
DEFINITIONS = ({"REQ":"SELECT systimestamp FROM dual",
"PARAM":"systimestamp",
"HELP":"System date"},
{"REQ": "SELECT banner FROM V$VERSION",
"PARAM": "banner",
"HELP": "Version(s)"},
{"REQ": "SELECT sys_context('USERENV', 'SERVER_HOST') FROM dual",
"PARAM": "SERVER_HOST",
"HELP": "Instance host name"},
{"REQ": "SELECT sys_context('USERENV', 'IP_ADDRESS') FROM dual",
"PARAM": "IP_ADDRESS",
"HELP": "IP address"},
{"REQ": "SELECT sys_context('USERENV', 'DB_NAME') FROM dual",
"PARAM": "DB_NAME",
"HELP": "Name of the database - db_name"},
{"REQ": "SELECT sys_context('USERENV', 'DB_UNIQUE_NAME') FROM dual",
"PARAM": "DB_UNIQUE_NAME",
"HELP": "Name of the database - db_unique_name"},
{"REQ": "SELECT sys_context('USERENV', 'INSTANCE_NAME') FROM dual",
"PARAM": "INSTANCE_NAME",
"HELP": "Instance name"},
{"REQ": "SELECT sys_context('USERENV', 'SERVICE_NAME') FROM dual",
"PARAM": "SERVICE_NAME",
"HELP": "Service Name (session)"},
{"REQ": "SELECT sys_context('USERENV', 'HOST') FROM dual",
"PARAM": "HOST",
"HELP": "Host machine name (client)"},
{"REQ": "SELECT sys_context('USERENV', 'OS_USER') FROM dual",
"PARAM": "OS_USER",
"HELP": "Username on client OS"},
{"REQ": "SELECT sys_context('USERENV', 'LANGUAGE') FROM dual",
"PARAM": "LANGUAGE",
"HELP": "Langage"},
{"REQ": "SELECT value FROM V$OPTION WHERE parameter = 'Oracle Database Vault'",
"PARAM": "VALUE",
"HELP": "Oracle Database Vault is enabled"},
{"REQ": "SELECT value FROM V$OPTION WHERE parameter = 'Java'",
"PARAM": "VALUE",
"HELP": "JAVA is enabled"},
{"REQ": "SELECT UPPER(VALUE) FROM V$PARAMETER WHERE UPPER(NAME) = 'AUDIT_SYS_OPERATIONS'",
"PARAM": "VALUE",
"HELP": "Audit is enabled for SYSDBA or SYSOPER"},
{"REQ": "SELECT UPPER(VALUE) FROM V$PARAMETER WHERE UPPER(NAME)='AUDIT_TRAIL'",
"PARAM": "VALUE",
"HELP": "Basic audit features level"},
{"REQ": "SELECT UPPER(VALUE) FROM V$PARAMETER WHERE UPPER(NAME)='SEC_CASE_SENSITIVE_LOGON'",
"PARAM": "VALUE",
"HELP": "Case-sensitivity is required for passwords"},
{"REQ": "SELECT UPPER(VALUE) FROM V$PARAMETER WHERE UPPER(NAME)='SEC_MAX_FAILED_LOGIN_ATTEMPTS'",
"PARAM": "VALUE",
"HELP": "Maximum failed login attemps (connection)"},
{"REQ": "SELECT PROFILE, RESOURCE_NAME, LIMIT FROM DBA_PROFILES WHERE RESOURCE_NAME='FAILED_LOGIN_ATTEMPTS'",
"HELP": "Maximum failed login attemps and after locked (days)",
"TABLE": True},
{"REQ": "SELECT PROFILE, RESOURCE_NAME, LIMIT FROM DBA_PROFILES WHERE RESOURCE_NAME='PASSWORD_LOCK_TIME'",
"HELP": "Password lock time (days)",
"TABLE": True},
{"REQ": "SELECT PROFILE, RESOURCE_NAME, LIMIT FROM DBA_PROFILES WHERE RESOURCE_NAME='PASSWORD_LIFE_TIME'",
"HELP": "Password life time (days)",
"TABLE": True},
{"REQ": "SELECT PROFILE, RESOURCE_NAME, LIMIT FROM DBA_PROFILES WHERE RESOURCE_NAME='PASSWORD_REUSE_MAX'",
"HELP": "Password reuse maximum (days)",
"TABLE": True},
{"REQ": "SELECT PROFILE, RESOURCE_NAME, LIMIT FROM DBA_PROFILES WHERE RESOURCE_NAME='PASSWORD_GRACE_TIME'",
"HELP": "Password grace time (days)",
"TABLE": True},
{"REQ": "SELECT PROFILE, RESOURCE_NAME, LIMIT FROM DBA_PROFILES WHERE RESOURCE_NAME='SESSIONS_PER_USER'",
"HELP": "Sessions per user",
"TABLE": True},
{"REQ": "SELECT * FROM USER_SYS_PRIVS",
"HELP": "Current user's privileges (via USER_SYS_PRIVS & no recursion on roles)",
"TABLE": True},
{"REQ": "SELECT * FROM USER_ROLE_PRIVS",
"HELP": "Current user's roles",
"TABLE": True},
{"REQ": "SELECT DISTINCT owner FROM all_tables",
"HELP": "Databases",
"TABLE":True},
{"REQ": "SELECT username FROM all_users",
"HELP": "All visible Users",
"TABLE": True},
{"REQ": "SELECT parameter, value FROM V$OPTION",
"HELP": "Database options and features",
"TABLE": True},
{"REQ": "SELECT * FROM DBA_REGISTRY",
"HELP": "Components loaded into the database",
"TABLE": True},
{"REQ": "select * from dba_registry_sqlpatch",
"HELP": "SQL patches installed (>= 12c)",
"TABLE": True},
{"REQ": "SELECT dbms_qopatch.GET_OPATCH_LIST FROM dual",
"HELP": "SQL patches installed with DBMS_QOPATCH (>= 12c)",
"TABLE": True},
{"REQ": "select * from sys.registry$history",
"HELP": "SQL patches installed (== 11g)",
"TABLE": True},
)
print("\n")
for aDefinition in DEFINITIONS:
if 'TABLE' in aDefinition and aDefinition['TABLE'] == True:
getColumnNames = True
ld = []
else:
getColumnNames = False
ld = [aDefinition["PARAM"]]
response = self.__execThisQuery__(query=aDefinition["REQ"],
ld=ld,
isquery=True,
getColumnNames=getColumnNames,
stringOnly=False)
if isinstance(response,Exception) :
logging.warning('Error with the SQL request {0}: {1}'.format(aDefinition["REQ"], str(response)))
print("--> {0}: unknown".format(aDefinition["HELP"]))
else:
if 'TABLE' in aDefinition and aDefinition['TABLE'] == True:
print("\n--> {0}:".format(aDefinition["HELP"]))
table = Texttable(max_width=getScreenSize()[0])
table.set_deco(Texttable.HEADER)
table.add_rows(response)
print(table.draw())
else:
for aRespone in response:
print("--> {0}: {1}".format(aDefinition["HELP"], aRespone[aDefinition["PARAM"]]))
def testAll (self):
'''
Test all functions
'''
self.args['print'].subtitle("Search in column names ?")
logging.info('Nothing to do, return True')
self.args['print'].goodNews("OK")
return True
def runSearchModule(args):
'''
Run the Search module
'''
status = True
if checkOptionsGivenByTheUser(args,["test-module","column-names","pwd-column-names","desc-tables","without-example","sql-shell",'basic-info']) == False : return EXIT_MISS_ARGUMENT
search = Search(args)
status = search.connection(stopIfError=True)
if args['test-module'] == True :
args['print'].title("Test if the Search module can be used")
status = search.testAll()
if args['basic-info'] == True :
args['print'].title("Test if the Search module can be used")
status = search.getBasicInformation(printStdout=True)
if ('column-names' in args)==True and args['column-names']!=None:
args['print'].title("Columns which contains the pattern '{0}'".format(args['column-names']))
table = search.searchInColumns(args['column-names'],showEmptyColumns=args['show-empty-columns'], withoutExample=args['without-example'])
if search.isEmptyTable(table) == True :
args['print'].badNews("no result found")
else :
args['print'].goodNews(table)
if args['pwd-column-names']==True:
args['print'].title("Columns which contains the pattern ~password~ like (multi language)")
table = search.searchPwdKeyworkInColumnNames(showEmptyColumns=args['show-empty-columns'], withoutExample=args['without-example'])
if search.isEmptyTable(table) == True :
args['print'].badNews("no result found")
else :
args['print'].goodNews(table)
if args['desc-tables']==True:
args['print'].title("Descibe each table which is accessible by the current user (without system tables)")
table = search.getDescOfEachNoSystemTable()
print(table)
if args['sql-shell'] == True:
args['print'].title("Starting an interactive SQL shell")
search.startInteractiveSQLShell()