tiredpixel/tunefl

View on GitHub
app/workers/compile_score_worker.rb

Summary

Maintainability
A
0 mins
Test Coverage
require 'tmpdir'
require 'erb'
require 'timeout'


class LilyPondException < StandardError; end


class CompileScoreWorker
  
  COMPILE_TIMEOUT = 60
  
  include Sidekiq::Worker
  
  def perform(id)
    score = Score.find(id)
    
    tmpdir = Dir.mktmpdir
    
    files = {
      :lilypond     => File.join(tmpdir, "#{id}.ly"),
      :midi         => File.join(tmpdir, "#{id}.midi"),
      :png          => File.join(tmpdir, "#{id}.png"),
      :preview_eps  => File.join(tmpdir, "#{id}.preview.eps"),
      :preview_png  => File.join(tmpdir, "#{id}.preview.png"),
    }
    
    begin
      self.class.generate_lilypond(score, files)
      
      Timeout::timeout(ENV['WORKER_COMPILE_TIMEOUT'].to_i || COMPILE_TIMEOUT) {
        self.class.compile_lilypond(score, files)
      }
      
      score.usable = true
      
      score.save
    rescue
      score.update_attribute(:usable, false)
      
      raise
    ensure
      FileUtils.remove_entry_secure tmpdir
    end
  end
  
  private
  
  def self.generate_lilypond(score, files)
    lilypond_data = ERB.new(File.read(
      Rails.root.join('app', 'views', 'erb', 'lilypond.ly.erb')
    )).result(binding)
    
    File.write(files[:lilypond], lilypond_data)
    
    score.lilypond = File.open(files[:lilypond])
  end
  
  def self.compile_lilypond(score, files)
    output = `lilypond -dsafe -dresolution=150 -dpreview --png --output #{File.dirname(files[:lilypond])} #{files[:lilypond]} 2>&1`
    
    raise LilyPondException, output unless $?.success?
    
    score.preview = File.open(files[:preview_png])
    
    score.midi = File.open(files[:midi])
  end
  
end