эффективное массовое обновление базы данных рельсов - PullRequest
5 голосов
/ 18 февраля 2010

Я пытаюсь создать утилиту rake, которая будет обновлять мою базу данных очень часто.

Это код, который у меня есть:

namespace :utils do

  # utils:update_ip
  # Downloads the file frim <url> to the temp folder then unzips it in <file_path>
  # Then updates the database.

  desc "Update ip-to-country database"
  task :update_ip => :environment do

    require 'open-uri'
    require 'zip/zipfilesystem'
    require 'csv'

    file_name = "ip-to-country.csv"
    file_path = "#{RAILS_ROOT}/db/" + file_name
    url = 'http://ip-to-country.webhosting.info/downloads/ip-to-country.csv.zip'


    #check last time we updated the database.
    mod_time = ''
    mod_time = File.new(file_path).mtime.httpdate    if File.exists? file_path

    begin
      puts 'Downloading update...'
      #send conditional GET to server
      zipped_file = open(url, {'If-Modified-Since' => mod_time})
    rescue OpenURI::HTTPError => the_error
      if the_error.io.status[0] == '304'
        puts 'Nothing to update.'
      else
        puts 'HTTPError: ' + the_error.message
      end
    else # file was downloaded without error.

      Rails.logger.info 'ip-to-coutry: Remote database was last updated: ' + zipped_file.meta['last-modified']
      delay = Time.now - zipped_file.last_modified
      Rails.logger.info "ip-to-country: Database was outdated for: #{delay} seconds (#{delay / 60 / 60 / 24 } days)"

      puts 'Unzipping...'
      File.delete(file_path) if File.exists? file_path
      Zip::ZipFile.open(zipped_file.path) do |zipfile|
        zipfile.extract(file_name, file_path)
      end

      Iptocs.delete_all

      puts "Importing new database..."


      # TODO: way, way too heavy find a better solution.


      CSV.open(file_path, 'r') do |row|
        ip = Iptocs.new(  :ip_from        => row.shift,
                        :ip_to          => row.shift,
                        :country_code2  => row.shift,
                        :country_code3  => row.shift,
                        :country_name   => row.shift)
        ip.save
      end #CSV
      puts "Complete."

    end #begin-resuce
  end #task
end #namespace

Проблема, с которой я столкнулся, заключается в том, что для ввода более 100 тысяч записей требуется несколько минут. Я хотел бы найти более эффективный способ обновления моей базы данных. В идеале это будет оставаться независимым от типа базы данных, но в противном случае мой рабочий сервер будет работать на MySQL.

Спасибо за понимание.

Ответы [ 5 ]

9 голосов
/ 18 февраля 2010

Вы пытались использовать AR Extensions для массового импорта? Вы получаете впечатляющие улучшения производительности, когда вы вставляете тысячи строк в БД. Посетите их веб-сайт для получения более подробной информации.

Обратитесь к этим примерам для получения дополнительной информации

Пример использования 1

Пример использования 2

Пример использования 3

3 голосов
/ 18 февраля 2010

Используйте утилиты уровня базы данных для высокой скорости Люк!

К сожалению, они специфичны для БД. Но они быстрые Для mysql, см. http://dev.mysql.com/doc/refman/5.1/en/load-data.html

1 голос
/ 02 мая 2012

Я сейчас экспериментирую с activerecord-import, который звучит очень многообещающе:

https://github.com/zdennis/activerecord-import

1 голос
/ 18 февраля 2010

Вы можете сгенерировать текстовый файл со всеми необходимыми вставками, а затем выполнить:

mysql -u user -p db_name < mytextfile.txt

Не уверен, что это будет быстрее, но стоит попробовать ...

0 голосов
/ 18 февраля 2010

Как говорит Ларри, используйте специальные утилиты импорта для БД, если файл имеет нужный формат. Однако если вам нужно манипулировать данными перед вставкой, вы можете сгенерировать один запрос INSERT с данными для многих строк, что быстрее, чем использование отдельного запроса для каждой строки (как это сделает ActiveRecord). Например:

INSERT INTO iptocs (ip_from, ip_to, country_code) VALUES
  ('xxx', 'xxx', 'xxx'),
  ('yyy', 'yyy', 'yyy'),
  ...;
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...