/python/lib/Lib/site-packages/django/db/backends/oracle/introspection.py

http://github.com/JetBrains/intellij-community · Python · 121 lines · 96 code · 12 blank · 13 comment · 13 complexity · 62fca0e91fd51d2af4483099144e8394 MD5 · raw file

  1. from django.db.backends import BaseDatabaseIntrospection
  2. import cx_Oracle
  3. import re
  4. foreign_key_re = re.compile(r"\sCONSTRAINT `[^`]*` FOREIGN KEY \(`([^`]*)`\) REFERENCES `([^`]*)` \(`([^`]*)`\)")
  5. class DatabaseIntrospection(BaseDatabaseIntrospection):
  6. # Maps type objects to Django Field types.
  7. data_types_reverse = {
  8. cx_Oracle.CLOB: 'TextField',
  9. cx_Oracle.DATETIME: 'DateField',
  10. cx_Oracle.FIXED_CHAR: 'CharField',
  11. cx_Oracle.NCLOB: 'TextField',
  12. cx_Oracle.NUMBER: 'DecimalField',
  13. cx_Oracle.STRING: 'CharField',
  14. cx_Oracle.TIMESTAMP: 'DateTimeField',
  15. }
  16. try:
  17. data_types_reverse[cx_Oracle.NATIVE_FLOAT] = 'FloatField'
  18. except AttributeError:
  19. pass
  20. try:
  21. data_types_reverse[cx_Oracle.UNICODE] = 'CharField'
  22. except AttributeError:
  23. pass
  24. def get_field_type(self, data_type, description):
  25. # If it's a NUMBER with scale == 0, consider it an IntegerField
  26. if data_type == cx_Oracle.NUMBER and description[5] == 0:
  27. if description[4] > 11:
  28. return 'BigIntegerField'
  29. else:
  30. return 'IntegerField'
  31. else:
  32. return super(DatabaseIntrospection, self).get_field_type(
  33. data_type, description)
  34. def get_table_list(self, cursor):
  35. "Returns a list of table names in the current database."
  36. cursor.execute("SELECT TABLE_NAME FROM USER_TABLES")
  37. return [row[0].lower() for row in cursor.fetchall()]
  38. def get_table_description(self, cursor, table_name):
  39. "Returns a description of the table, with the DB-API cursor.description interface."
  40. cursor.execute("SELECT * FROM %s WHERE ROWNUM < 2" % self.connection.ops.quote_name(table_name))
  41. description = []
  42. for desc in cursor.description:
  43. description.append((desc[0].lower(),) + desc[1:])
  44. return description
  45. def table_name_converter(self, name):
  46. "Table name comparison is case insensitive under Oracle"
  47. return name.lower()
  48. def _name_to_index(self, cursor, table_name):
  49. """
  50. Returns a dictionary of {field_name: field_index} for the given table.
  51. Indexes are 0-based.
  52. """
  53. return dict([(d[0], i) for i, d in enumerate(self.get_table_description(cursor, table_name))])
  54. def get_relations(self, cursor, table_name):
  55. """
  56. Returns a dictionary of {field_index: (field_index_other_table, other_table)}
  57. representing all relationships to the given table. Indexes are 0-based.
  58. """
  59. cursor.execute("""
  60. SELECT ta.column_id - 1, tb.table_name, tb.column_id - 1
  61. FROM user_constraints, USER_CONS_COLUMNS ca, USER_CONS_COLUMNS cb,
  62. user_tab_cols ta, user_tab_cols tb
  63. WHERE user_constraints.table_name = %s AND
  64. ta.table_name = %s AND
  65. ta.column_name = ca.column_name AND
  66. ca.table_name = %s AND
  67. user_constraints.constraint_name = ca.constraint_name AND
  68. user_constraints.r_constraint_name = cb.constraint_name AND
  69. cb.table_name = tb.table_name AND
  70. cb.column_name = tb.column_name AND
  71. ca.position = cb.position""", [table_name, table_name, table_name])
  72. relations = {}
  73. for row in cursor.fetchall():
  74. relations[row[0]] = (row[2], row[1])
  75. return relations
  76. def get_indexes(self, cursor, table_name):
  77. """
  78. Returns a dictionary of fieldname -> infodict for the given table,
  79. where each infodict is in the format:
  80. {'primary_key': boolean representing whether it's the primary key,
  81. 'unique': boolean representing whether it's a unique index}
  82. """
  83. # This query retrieves each index on the given table, including the
  84. # first associated field name
  85. # "We were in the nick of time; you were in great peril!"
  86. sql = """\
  87. SELECT LOWER(all_tab_cols.column_name) AS column_name,
  88. CASE user_constraints.constraint_type
  89. WHEN 'P' THEN 1 ELSE 0
  90. END AS is_primary_key,
  91. CASE user_indexes.uniqueness
  92. WHEN 'UNIQUE' THEN 1 ELSE 0
  93. END AS is_unique
  94. FROM all_tab_cols, user_cons_columns, user_constraints, user_ind_columns, user_indexes
  95. WHERE all_tab_cols.column_name = user_cons_columns.column_name (+)
  96. AND all_tab_cols.table_name = user_cons_columns.table_name (+)
  97. AND user_cons_columns.constraint_name = user_constraints.constraint_name (+)
  98. AND user_constraints.constraint_type (+) = 'P'
  99. AND user_ind_columns.column_name (+) = all_tab_cols.column_name
  100. AND user_ind_columns.table_name (+) = all_tab_cols.table_name
  101. AND user_indexes.uniqueness (+) = 'UNIQUE'
  102. AND user_indexes.index_name (+) = user_ind_columns.index_name
  103. AND all_tab_cols.table_name = UPPER(%s)
  104. """
  105. cursor.execute(sql, [table_name])
  106. indexes = {}
  107. for row in cursor.fetchall():
  108. indexes[row[0]] = {'primary_key': row[1], 'unique': row[2]}
  109. return indexes