]> source.dussan.org Git - sonarqube.git/blob
9db891c448232f12dd9ba9fa8bf708b70b281eed
[sonarqube.git] /
1 require 'active_record/connection_adapters/abstract/schema_definitions'
2
3 module ::JdbcSpec
4   # Don't need to load native mysql adapter
5   $LOADED_FEATURES << "active_record/connection_adapters/mysql_adapter.rb"
6
7   module ActiveRecordExtensions
8     def mysql_connection(config)
9       config[:port] ||= 3306
10       if config[:url]
11         config[:url] = config[:url]['?'] ? "#{config[:url]}&#{MySQL::URL_OPTIONS}" : "#{config[:url]}?#{MySQL::URL_OPTIONS}"
12       else
13         config[:url] = "jdbc:mysql://#{config[:host]}:#{config[:port]}/#{config[:database]}?#{MySQL::URL_OPTIONS}"
14       end
15       config[:driver] = "com.mysql.jdbc.Driver"
16       jdbc_connection(config)
17     end
18   end
19
20   module MySQL
21     URL_OPTIONS = "zeroDateTimeBehavior=convertToNull&jdbcCompliantTruncation=false&useUnicode=true&characterEncoding=utf8"
22     def self.column_selector
23       [/mysql/i, lambda {|cfg,col| col.extend(::JdbcSpec::MySQL::Column)}]
24     end
25
26     def self.adapter_selector
27       [/mysql/i, lambda {|cfg,adapt| adapt.extend(::JdbcSpec::MySQL)}]
28     end
29
30     def self.extended(adapter)
31       adapter.execute("SET SQL_AUTO_IS_NULL=0")
32     end
33
34     module Column
35       TYPES_ALLOWING_EMPTY_STRING_DEFAULT = Set.new([:binary, :string, :text])
36
37       def simplified_type(field_type)
38         return :boolean if field_type =~ /tinyint\(1\)|bit/i
39         return :string  if field_type =~ /enum/i
40         super
41       end
42
43       def init_column(name, default, *args)
44         @original_default = default
45         @default = nil if missing_default_forged_as_empty_string?
46       end
47
48       # MySQL misreports NOT NULL column default when none is given.
49       # We can't detect this for columns which may have a legitimate ''
50       # default (string, text, binary) but we can for others (integer,
51       # datetime, boolean, and the rest).
52       #
53       # Test whether the column has default '', is not null, and is not
54       # a type allowing default ''.
55       def missing_default_forged_as_empty_string?
56         !null && @original_default == '' && !TYPES_ALLOWING_EMPTY_STRING_DEFAULT.include?(type)
57       end
58     end
59
60     def modify_types(tp)
61       tp[:primary_key] = "int(11) DEFAULT NULL auto_increment PRIMARY KEY"
62       tp[:decimal] = { :name => "decimal" }
63       tp[:timestamp] = { :name => "datetime" }
64       tp[:datetime][:limit] = nil
65       tp
66     end
67
68     # QUOTING ==================================================
69
70     def quote(value, column = nil)
71       return value.quoted_id if value.respond_to?(:quoted_id)
72
73       if column && column.type == :primary_key
74         value.to_s
75       elsif column && String === value && column.type == :binary && column.class.respond_to?(:string_to_binary)
76         s = column.class.string_to_binary(value).unpack("H*")[0]
77         "x'#{s}'"
78       elsif BigDecimal === value
79         "'#{value.to_s("F")}'"
80       else
81         super
82       end
83     end
84
85     def quote_column_name(name) #:nodoc:
86         "`#{name}`"
87     end
88
89     def quote_table_name(name) #:nodoc:
90       quote_column_name(name).gsub('.', '`.`')
91     end
92
93     def quoted_true
94         "1"
95     end
96
97     def quoted_false
98         "0"
99     end
100
101     def begin_db_transaction #:nodoc:
102       @connection.begin
103     rescue Exception
104       # Transactions aren't supported
105     end
106
107     def commit_db_transaction #:nodoc:
108       @connection.commit
109     rescue Exception
110       # Transactions aren't supported
111     end
112
113     def rollback_db_transaction #:nodoc:
114       @connection.rollback
115     rescue Exception
116       # Transactions aren't supported
117     end
118
119     def disable_referential_integrity(&block) #:nodoc:
120       old = select_value("SELECT @@FOREIGN_KEY_CHECKS")
121       begin
122         update("SET FOREIGN_KEY_CHECKS = 0")
123         yield
124       ensure
125         update("SET FOREIGN_KEY_CHECKS = #{old}")
126       end
127     end
128
129     # SCHEMA STATEMENTS ========================================
130
131     def structure_dump #:nodoc:
132       if supports_views?
133         sql = "SHOW FULL TABLES WHERE Table_type = 'BASE TABLE'"
134       else
135         sql = "SHOW TABLES"
136       end
137
138       select_all(sql).inject("") do |structure, table|
139         table.delete('Table_type')
140
141         hash = select_one("SHOW CREATE TABLE #{quote_table_name(table.to_a.first.last)}")
142
143         if(table = hash["Create Table"])
144           structure += table + ";\n\n"
145         elsif(view = hash["Create View"])
146           structure += view + ";\n\n"
147         end
148       end
149     end
150
151     def recreate_database(name) #:nodoc:
152       drop_database(name)
153       create_database(name)
154     end
155
156     def create_database(name, options = {}) #:nodoc:
157       if options[:collation]
158         execute "CREATE DATABASE `#{name}` DEFAULT CHARACTER SET `#{options[:charset] || 'utf8'}` COLLATE `#{options[:collation]}`"
159       else
160         execute "CREATE DATABASE `#{name}` DEFAULT CHARACTER SET `#{options[:charset] || 'utf8'}`"
161       end
162     end
163
164     def drop_database(name) #:nodoc:
165       execute "DROP DATABASE IF EXISTS `#{name}`"
166     end
167
168     def current_database
169       select_one("SELECT DATABASE() as db")["db"]
170     end
171
172     def create_table(name, options = {}) #:nodoc:
173       super(name, {:options => "ENGINE=InnoDB CHARACTER SET utf8 COLLATE utf8_bin"}.merge(options))
174     end
175
176     def rename_table(name, new_name)
177       execute "RENAME TABLE #{quote_table_name(name)} TO #{quote_table_name(new_name)}"
178     end
179
180     def change_column_default(table_name, column_name, default) #:nodoc:
181       current_type = select_one("SHOW COLUMNS FROM #{quote_table_name(table_name)} LIKE '#{column_name}'")["Type"]
182
183       execute("ALTER TABLE #{quote_table_name(table_name)} CHANGE #{quote_column_name(column_name)} #{quote_column_name(column_name)} #{current_type} DEFAULT #{quote(default)}")
184     end
185
186     def change_column(table_name, column_name, type, options = {}) #:nodoc:
187       unless options_include_default?(options)
188         if column = columns(table_name).find { |c| c.name == column_name.to_s }
189           options[:default] = column.default
190         else
191           raise "No such column: #{table_name}.#{column_name}"
192         end
193       end
194
195       change_column_sql = "ALTER TABLE #{quote_table_name(table_name)} CHANGE #{quote_column_name(column_name)} #{quote_column_name(column_name)} #{type_to_sql(type, options[:limit], options[:precision], options[:scale])}"
196       add_column_options!(change_column_sql, options)
197       execute(change_column_sql)
198     end
199
200     def rename_column(table_name, column_name, new_column_name) #:nodoc:
201       cols = select_one("SHOW COLUMNS FROM #{quote_table_name(table_name)} LIKE '#{column_name}'")
202       current_type = cols["Type"] || cols["COLUMN_TYPE"]
203       execute "ALTER TABLE #{quote_table_name(table_name)} CHANGE #{quote_table_name(column_name)} #{quote_column_name(new_column_name)} #{current_type}"
204     end
205
206     def add_limit_offset!(sql, options) #:nodoc:
207       if limit = options[:limit]
208         unless offset = options[:offset]
209           sql << " LIMIT #{limit}"
210         else
211           sql << " LIMIT #{offset}, #{limit}"
212         end
213       end
214     end
215
216     def show_variable(var)
217       res = execute("show variables like '#{var}'")
218       row = res.detect {|row| row["Variable_name"] == var }
219       row && row["Value"]
220     end
221
222     def charset
223       show_variable("character_set_database")
224     end
225
226     def collation
227       show_variable("collation_database")
228     end
229
230     private
231     def supports_views?
232       false
233     end
234   end
235 end