canvas-lms/lib/cuty_capt.rb

162 lines
5.1 KiB
Ruby

#
# Copyright (C) 2012 Instructure, Inc.
#
# This file is part of Canvas.
#
# Canvas is free software: you can redistribute it and/or modify it under
# the terms of the GNU Affero General Public License as published by the Free
# Software Foundation, version 3 of the License.
#
# Canvas is distributed in the hope that it will be useful, but WITHOUT ANY
# WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
# A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
# details.
#
# You should have received a copy of the GNU Affero General Public License along
# with this program. If not, see <http://www.gnu.org/licenses/>.
#
# A small wrapper around the CutyCapt binary.
#
# Requires a config file in RAILS_ROOT/config/cutycapt.yml that looks like this:
#
# production:
# path: /usr/bin/cutycapt
# delay: 3000
# timeout: 30000
# display: ':0'
#
# delay is how many ms to wait before taking the snapshot (to let the page finish rendering)
# display is whatever display cutycapt should use. (You should probably use Xvfb.)
require 'resolv'
require 'netaddr'
require 'action_controller_test_process'
class CutyCapt
CUTYCAPT_DEFAULTS = {
:delay => 3000,
:timeout => 60000,
:ip_blacklist => [ '10.0.0.0/8', '172.16.0.0/12', '192.168.0.0/16', '169.254.169.254' ],
:domain_blacklist => [ ],
:allowed_schemes => [ 'http', 'https' ],
:lang => 'en,*;q=0.9'
}
cattr_writer :config
def self.config
return @@config if defined?(@@config) && @@config
setting = (ConfigFile.load('cutycapt') || {}).symbolize_keys
@@config = CUTYCAPT_DEFAULTS.merge(setting).with_indifferent_access
self.process_config
@@config = nil unless @@config[:path]
@@config
end
def self.process_config
@@config[:ip_blacklist] = @@config[:ip_blacklist].map {|ip| NetAddr::CIDR.create(ip) } if @@config[:ip_blacklist]
@@config[:domain_blacklist] = @@config[:domain_blacklist].map {|domain| Resolv::DNS::Name.create(domain) } if @@config[:domain_blacklist]
end
def self.logger
Rails.logger
end
def self.enabled?
return !self.config.nil?
end
def self.verify_url(url)
config = self.config
uri = URI.parse(url)
unless config[:allowed_schemes] && config[:allowed_schemes].include?(uri.scheme)
logger.warn("Skipping non-http[s] URL: #{url}")
return false
end
dns_host = Resolv::DNS::Name.create(uri.host)
if config[:domain_blacklist] && config[:domain_blacklist].any? {|bl_host| dns_host == bl_host || dns_host.subdomain_of?(bl_host) }
logger.warn("Skipping url because of blacklisted domain: #{url}")
return false
end
addresses = Resolv.getaddresses(uri.host)
if config[:ip_blacklist] && addresses.any? {|address| config[:ip_blacklist].any? {|cidr| cidr.matches?(address) rescue false } }
logger.warn("Skipping url because of blacklisted IP address: #{url}")
return false
end
true
end
def self.cuty_arguments(path, url, img_file, format, delay, timeout, lang)
[ path, "--url=#{url}", "--out=#{img_file}", "--out-format=#{format}", "--delay=#{delay}", "--max-wait=#{timeout}", "--header=Accept-Language:#{lang}" ]
end
def self.snapshot_url(url, format = "png", &block)
return nil unless config = self.config
return nil unless self.verify_url(url)
tmp_file = Tempfile.new(['websnappr', ".#{format}"])
img_file = tmp_file.path
# We need to finalize the tmp_file now, because if we don't then it will get closed
# in the child process below, deleting it. This does introduce a potential race condition
# but in practice shouldn't be a problem since Tempfiles normally include the process pid.
tmp_file.close!
success = true
start = Time.now
logger.info("Starting web capture of #{url}")
if (pid = fork).nil?
ENV["DISPLAY"] = config[:display] if config[:display]
Kernel.exec(*cuty_arguments(config[:path], url, img_file, format, config[:delay], config[:timeout], config[:lang]))
else
begin
Timeout::timeout(config[:timeout].to_i / 1000) do
Process.waitpid(pid)
unless $?.success?
logger.error("Capture failed with code: #{$?.exitstatus}")
success = false
end
end
rescue Timeout::Error
logger.error("Capture timed out")
Process.kill("KILL", pid)
Process.waitpid(pid)
success = false
end
end
if !success
File.unlink(img_file) if File.exists?(img_file)
return nil
else
logger.info("Capture took #{Time.now.to_i - start.to_i} seconds")
end
if block_given?
yield img_file
File.unlink(img_file) if File.exists?(img_file)
return nil
end
img_file
end
def self.snapshot_attachment_for_url(url)
require 'action_controller_test_process'
attachment = nil
self.snapshot_url(url, "png") do |file_path|
# this is a really odd way to get Attachment the data it needs, which
# should probably be remedied at some point
attachment = Attachment.new(:uploaded_data => Rack::Test::UploadedFile.new(file_path, "image/png"))
end
return attachment
end
end