require 'rubygems' gem 'RequirePaths'; require 'require_paths' require_paths '.','..' require 'fileutils' require 'net/smtp' require 'ihl_ruby/misc_utils' require 'ihl_ruby/logging' require 'ihl_ruby/string_utils' require 'ihl_ruby/xml_utils' require 'ihl_ruby/extend_base_classes' require 'ihl_ruby/shell_extras' require 'ihl_ruby/config' require 'ihl_ruby/database_utils' THIS_FILE = File.expand_path(__FILE__) THIS_DIR = File.dirname(THIS_FILE) module YoreCore class KeepDaily attr_reader :keep_age def initialize(aKeepAge=14) @keep_age = aKeepAge end def is? true end def age(aDate) end def keep?(aDate) end end class KeepWeekly attr_reader :keep_age def initialize(aKeepAge=14) @keep_age = aKeepAge end def is? end def age(aDate) end def keep?(aDate) end end class KeepMonthly attr_reader :keep_age def initialize(aKeepAge=14) @keep_age = aKeepAge end def is? end def age(aDate) end def keep?(aDate) end end class Yore DEFAULT_CONFIG = { :kind => '', :basepath => '', :keep_daily => 14, :keep_weekly => 12, :keep_monthly => 12, :crypto_iv => "3A63775C1E3F291B0925578165EB917E", # apparently a string of up to 32 random hex digits :crypto_key => "07692FC8656F04AE5518B80D38681E038A3C12050DF6CC97CEEC33D800D5E2FE", # apparently a string of up to 64 random hex digits :first_hour => 4, :prefix => 'backup', :log_level => 'INFO', :bucket => '', :email_report => false, :mail_host => '', :mail_port => 25, :mail_helodomain => '', :mail_user => '', :mail_password => '', :mail_from => '', :mail_from_alias => '', :mail_to => '', :mail_to_alias => '', :mail_auth => :plain, :mysqldump => 'mysqldump', :RAILS_ENV => '' } attr_reader :config attr_reader :logger attr_reader :reporter attr_reader :keepers def initialize(aConfig=nil) DEFAULT_CONFIG[:email_report] = false # fixes some bug where this was nil cons = ConsoleLogger.new() cons.level = Logger::Severity.const_get(config[:log_level]) rescue Logger::Severity::INFO report_file = MiscUtils::temp_file @reporter = Logger.new(report_file) @reporter.formatter = ConsoleLogger::ReportFormatter.new @reporter.level = cons.level @logger = MultiLogger.new([cons,@reporter]) @logger.info "Yore file and database backup tool for Amazon S3 " @logger.info "(c) 2009 Buzzware Solutions (www.buzzware.com.au)" @logger.info "-------------------------------------------------" @logger.info "" @logger.info "report file: #{report_file}" configure(aConfig) end #aOptions may require {:basepath => File.dirname(File.expand_path(job))} def self.launch(aConfigXml,aCmdOptions=nil,aOptions=nil) result = Yore.new() result.configure(aConfigXml,aCmdOptions,aOptions) return result end def create_empty_config_xml() s = <<-EOS EOS xdoc = REXML::Document.new(s) return xdoc.root end def get_rails_db_details(aRailsPath,aRailsEnv) return nil unless aRailsPath && aRailsEnv && aRailsEnv!='' return nil unless dbyml = (YAML::load(File.open(File.expand_path('config/database.yml',aRailsPath))) rescue nil) return dbyml[aRailsEnv] && dbyml[aRailsEnv].symbolize_keys end def expand_app_option(kind=nil) kind = config[:kind] unless kind && !kind.empty? return nil unless kind && !kind.empty? config.xmlRoot = create_empty_config_xml() if !config.xmlRoot case kind when 'spree' # add file source xmlSources = XmlUtils.single_node(config.xmlRoot,'/Yore/Sources') if xmlSources strSource = <<-EOS public/assets/products EOS XmlUtils.add_xml_from_string(strSource,xmlSources) end expand_app_option('rails') # do again when 'rails' # * add db source from database.yml # load database from config[:basepath],'config/database.yml' #if (dbyml = YAML::load(File.open(File.expand_path('config/database.yml',config[:basepath]))) rescue nil) # if env = (config[:RAILS_ENV] && config[:RAILS_ENV]!='' && config[:RAILS_ENV]) # if (db_details = dbyml[env]) && db_details = get_rails_db_details(config[:basepath],config[:RAILS_ENV]) xmlSources = XmlUtils.single_node(config.xmlRoot,'/Yore/Sources') if db_details && xmlSources strSource = <<-EOS rails_app.sql EOS XmlUtils.add_xml_from_string(strSource,xmlSources) end end end # read the config however its given and return a hash with values in their correct type, and either valid or nil # keys must be :symbols for aOptions. aConfig and aCmdOptions can be strings def configure(aConfig,aCmdOptions = nil,aOptions = nil) config_to_read = {} if aConfig.is_a?(String) aConfig = File.expand_path(aConfig) logger.info "Job file: #{aConfig}" op = {:basepath => File.dirname(aConfig)} xml = XmlUtils.get_file_root(aConfig) return configure(xml,aCmdOptions,op) end if @config config_as_hash = nil case aConfig when nil then ; # do nothing when Hash,::ConfigClass then config_as_hash = aConfig when REXML::Element then config_as_hash = XmlUtils.read_simple_items(aConfig,'/Yore/SimpleItems') config.xmlRoot = aConfig # overwriting previous! perhaps should merge else raise StandardError.new('unsupported type') end config_as_hash.each{|n,v| config_to_read[n.to_sym] = v} if config_as_hash # merge given new values else @config = ConfigXmlClass.new(DEFAULT_CONFIG,aConfig) end aCmdOptions.each{|k,v| config_to_read[k.to_sym] = v} if aCmdOptions # merge command options config_to_read.merge!(aOptions) if aOptions # merge options config.read(config_to_read) config[:basepath] = File.expand_path(Dir.pwd) if !config[:basepath] || config[:basepath]=='' expand_app_option() @keepers = Array.new @keepers << KeepDaily.new(config[:keep_daily]) @keepers << KeepWeekly.new(config[:keep_weekly]) @keepers << KeepMonthly.new(config[:keep_monthly]) end def do_action(aAction,aArgs,aCmdOptions) logger.info "Executing command: #{aAction} ...\n" begin send(aAction,aArgs,aCmdOptions) rescue Exception => e logger.info {e.backtrace.join("\n")} logger.warn "#{e.class.to_s}: during #{aAction.to_s}(#{(aArgs && aArgs.inspect).to_s}): #{e.message.to_s}" end end def shell(aCommandline,&aBlock) logger.debug "To shell: " + aCommandline result = block_given? ? POpen4::shell(aCommandline,nil,nil,&aBlock) : POpen4::shell(aCommandline) logger.debug "From shell: '#{result.inspect}'" return result[:stdout] end def s3shell(aCommandline) shell(aCommandline) do |r| r[:exitcode] = 1 if r[:stderr].length > 0 end end def get_log logger.close # read in log and return end def get_report MiscUtils::string_from_file(@reporter.logdev.filename) end def temp_path @temp_path = MiscUtils.make_temp_dir('yore') unless @temp_path return @temp_path end def self.filemap_from_filelist(aFiles) ancestor_path = MiscUtils.file_list_ancestor(aFiles) filemap = {} aFiles.each do |fp| filemap[fp] = MiscUtils.path_debase(fp,ancestor_path) end filemap end def keep_file?(aFile) end # By default, GNU tar suppresses a leading slash on absolute pathnames while creating or reading a tar archive. (You can suppress this with the -p option.) # tar : http://my.safaribooksonline.com/0596102461/I_0596102461_CHP_3_SECT_9#snippet # get files from wherever they are into a single file def compress(aSourceFiles,aDestFile,aParentDir=nil) logger.info "Collecting files ..." #logger.info aSourceFiles.join("\n") #filelist = filemap = nil #if aSourceFiles.is_a?(Hash) # filelist = aSourceFiles.keys # filemap = aSourceFiles #else # assume array # filelist = aSourceFiles # filemap = Yore.filemap_from_filelist(aSourceFiles) #end #aParentDir ||= MiscUtils.file_list_ancestor(filelist) listfile = MiscUtils.temp_file MiscUtils.string_to_file( aSourceFiles.join("\n"), #filelist.sort.map{|p| MiscUtils.path_debase(p, aParentDir)}.join("\n"), listfile ) tarfile = MiscUtils.file_change_ext(aDestFile, 'tar') shell("tar cv #{aParentDir ? '--directory='+aParentDir.to_s : ''} --file=#{tarfile} --files-from=#{listfile}") logger.info "Compressing ..." tarfile_size = File.size(tarfile) shell("bzip2 #{tarfile}; mv #{tarfile}.bz2 #{aDestFile}") logger.info "Compressed #{'%.1f' % (tarfile_size*1.0/2**10)} KB to #{'%.1f' % (File.size(aDestFile)*1.0/2**10)} KB" end def uncompress(aArchive,aDestination=nil,aArchiveContent=nil) #tarfile = File.expand_path(MiscUtils.file_change_ext(File.basename(aArchive),'tar'),temp_dir) #shell("bunzip2 #{tarfile}; mv #{tarfile}.bz2 #{aDestFile}") # #shell("tar cv #{aParentDir ? '--directory='+aParentDir.to_s : ''} --file=#{tarfile} --files-from=#{listfile}") #logger.info "Compressing ..." #tarfile_size = File.size(tarfile) #shell("bzip2 #{tarfile}; mv #{tarfile}.bz2 #{aDestFile}") #logger.info "Compressed #{'%.1f' % (tarfile_size*1.0/2**10)} KB to #{'%.1f' % (File.size(aDestFile)*1.0/2**10)} KB" aDestination ||= MiscUtils.make_temp_dir('uncompress') FileUtils.mkdir_p(aDestination) shell("tar xvf #{aArchive} #{aArchiveContent.to_s} --directory=#{aDestination} --bzip2") end def pack(aFileIn,aFileOut) logger.info "Encrypting ..." shell "openssl enc -aes-256-cbc -K #{config[:crypto_key]} -iv #{config[:crypto_iv]} -in #{aFileIn} -out #{aFileOut}" end def unpack(aFileIn,aFileOut) shell "openssl enc -d -aes-256-cbc -K #{config[:crypto_key]} -iv #{config[:crypto_iv]} -in #{aFileIn} -out #{aFileOut}" end def ensure_bucket(aBucket=nil) aBucket ||= config[:bucket] logger.info "Ensuring S3 bucket #{aBucket} exists ..." s3shell "s3cmd createbucket #{aBucket}" end # uploads the given file to the current bucket as its basename def upload(aFile) #ensure_bucket() logger.info "Uploading #{File.basename(aFile)} to S3 bucket #{config[:bucket]} ..." s3shell "s3cmd put #{config[:bucket]}:#{File.basename(aFile)} #{aFile}" end # downloads the given file from the current bucket as its basename def download(aFile) s3shell "s3cmd get #{config[:bucket]}:#{File.basename(aFile)} #{aFile}" end # calculate the date (with no time component) based on :day_begins_hour and the local time def backup_date(aTime) (aTime.localtime - (config[:first_hour]*3600)).date end # generates filename based on date and config # config : # :first_hour # : def encode_file_name(aTimeNow=Time.now) "#{config[:prefix]}-#{backup_date(aTimeNow).date_numeric}.yor" end # return date based on filename def decode_file_name(aFilename) prefix,date,ext = aFilename.scan(/(.*?)\-(.*?)\.(.*)/).flatten return Time.from_date_numeric(date) end def clean end # "/usr/bin/env" sets normal vars # eg. 30 14 * * * /usr/bin/env ruby /Users/kip/svn/thewall/script/runner /Users/kip/svn/thewall/app/delete_old_posts.rb # http://www.ameravant.com/posts/recurring-tasks-in-ruby-on-rails-using-runner-and-cron-jobs # install gems # make folder with correct folder structure # copy in files # add to crontab, with just email sending, then call backup def report return unless config[:email_report] msg = get_report() logger.info "Sending report via email to #{config[:mail_to]} ..." MiscUtils::send_email( :host => config[:mail_host], :port => config[:mail_port], :helodomain => config[:mail_helodomain], :user => config[:mail_user], :password => config[:mail_password], :from => config[:mail_from], :from_alias => config[:mail_from_alias], :to => config[:mail_to], :to_alias => config[:mail_to_alias], :auth => config[:mail_auth], :subject => 'backup report', :message => msg ) end def self.database_from_xml(aDatabaseNode) return { :host => aDatabaseNode.attributes['Host'], :username => aDatabaseNode.attributes['User'], :password => aDatabaseNode.attributes['Password'], :database => aDatabaseNode.attributes['Name'], :file => XmlUtils::peek_node_value(aDatabaseNode, "ToFile"), :archive_file => XmlUtils::peek_node_value(aDatabaseNode, "ArchiveFile") } end def collect_file_list(aSourcesXml,aTempFolder) filelist = [] sourceFound = false if aSourcesXml REXML::XPath.each(aSourcesXml,'Source') do |xmlSource| case xmlSource.attributes['Type'] when 'File' then # BasePath tag provides base path for IncludePaths to be relative to. Also indicates root folder for archive bp = MiscUtils.path_combine(config[:basepath],XmlUtils::peek_node_value(xmlSource, "@BasePath")) filelist << '-C'+bp REXML::XPath.each(xmlSource, 'IncludePath') do |xmlPath| files = MiscUtils::recursive_file_list(MiscUtils::path_combine(bp,xmlPath.text)) files.map!{|f| MiscUtils.path_debase(f,bp)} filelist += files sourceFound = true end when 'MySql' then # # # ~/dbdump.sql # # REXML::XPath.each(xmlSource, 'Database') do |xmlDb| args = Yore::database_from_xml(xmlDb) file = args.delete(:file) #legacy, absolute path arc_file = args.delete(:archive_file) #path in archive unless args[:host] && args[:username] && args[:password] && args[:database] && (file||arc_file) raise StandardError.new("Invalid or missing parameter") end if arc_file arc_file = MiscUtils.path_debase(arc_file,'/') sql_file = File.expand_path(arc_file,aTempFolder) FileUtils.mkdir_p(File.dirname(sql_file)) # create folders as necessry DatabaseUtils::save_database(args,sql_file) #db_to_file(args,sql_file) filelist << '-C'+aTempFolder filelist << arc_file sourceFound = true else DatabaseUtils::save_database(args,sql_file) filelist << file sourceFound = true end end end end end raise StandardError.new("Backup source found but file list empty") if sourceFound && filelist.empty? return filelist end def rails_tmp_path return @rails_tmp_path if @rails_tmp_path @rails_tmp_path = File.join(config[:basepath],'tmp/yore',Time.now.strftime('%Y%m%d-%H%M%S')) end def self.move_folder(aPath1,aPath2) path2Parent = MiscUtils.path_parent(aPath2) FileUtils.mkdir_p(path2Parent) FileUtils.mv(aPath1, path2Parent, :force => true) end def self.copy_folder(aPath1,aPath2) path2Parent = MiscUtils.path_parent(aPath2) FileUtils.mkdir_p(path2Parent) FileUtils.cp_r(aPath1, path2Parent) end def save_internal(aFilename) FileUtils.mkdir_p(files_path = File.join(temp_path,'files')) filelist = collect_file_list(XmlUtils.single_node(config.xmlRoot,'/Yore/Sources'),files_path) compress(filelist,aFilename) end # # ACTIONS # def save(aArgs,aCmdOptions=nil) fnArchive = aArgs.is_a?(Array) ? aArgs.first : aArgs #only supported argument configure(nil,aCmdOptions) save_internal(fnArchive) end def backup(aArgs,aCmdOptions=nil) # was aJobFiles unless aCmdOptions && aCmdOptions[:config] # assume already configured if config option specified, but back supports first arg being config file job = aArgs.first configure(job,aCmdOptions || {}) end temp_file = File.expand_path('backup.tar',temp_path) save_internal(temp_file) backup_file = File.expand_path(encode_file_name(),temp_path) pack(temp_file,backup_file) upload(backup_file) #clean end def load(aArgs,aCmdOptions=nil) fnArchive = aArgs.is_a?(Array) ? aArgs.first : aArgs #only supported argument configure(nil,aCmdOptions) FileUtils.mkdir_p(archive_path = File.join(temp_path,'archive')) uncompress(fnArchive,archive_path) xmlSources = XmlUtils.single_node(config.xmlRoot,'/Yore/Sources') REXML::XPath.each(xmlSources,'Source') do |xmlSource| case xmlSource.attributes['Type'] when 'File' then # # public/assets/products # REXML::XPath.each(xmlSource,'IncludePath') do |xmlIncludePath| pathArchive = xmlIncludePath.text() pathUncompressed = File.join(archive_path,pathArchive) pathTmp = File.join(rails_tmp_path,pathArchive) pathDest = File.join(config[:basepath],pathArchive) # move basepath/relativepath to tmp/yore/090807-010203/relativepath Yore::move_folder(pathDest,pathTmp) if File.exists?(pathDest) # get and copy to basepath/relativepath Yore::copy_folder(pathUncompressed,pathDest) if File.exists?(pathUncompressed) end when 'MySql' then db_details = Yore::database_from_xml(XmlUtils.single_node(xmlSource,'Database')) DatabaseUtils.load_database(db_details,File.join(archive_path,db_details[:archive_file])) end end end def test_email(*aDb) args = { :host => config[:mail_host], :port => config[:mail_port], :helodomain => config[:mail_helodomain], :user => config[:mail_user], :password => config[:mail_password], :from => config[:mail_from], :from_alias => config[:mail_from_alias], :to => config[:mail_to], :to_alias => config[:mail_to_alias], :auth => config[:mail_auth], :subject => 'email test', :message => 'Just testing email sending' } logger.debug args.inspect MiscUtils::send_email(args) end end end