344 lines
10 KiB
Ruby
344 lines
10 KiB
Ruby
require 'fileutils'
|
|
require 'zip/zip'
|
|
|
|
# Just a simple helper class to export quickly an existing and live locomotive website.
|
|
# FIXME: will be replaced by the API in the next months
|
|
module Locomotive
|
|
class Export
|
|
|
|
@@instance = nil
|
|
|
|
def initialize(site, filename = nil)
|
|
@site = site
|
|
@filename = filename || Time.now.to_i.to_s
|
|
@target_folder = File.join(Rails.root, 'tmp', 'export', @filename)
|
|
@site_hash = {} # used to generate the site.yml and compiled_site.yml files
|
|
|
|
self.create_target_folder
|
|
end
|
|
|
|
def run!
|
|
self.initialize_site_hash
|
|
|
|
self.log('copying content assets')
|
|
self.copy_content_assets
|
|
|
|
self.log('copying theme assets')
|
|
self.copy_theme_assets
|
|
|
|
self.log('copying pages')
|
|
self.copy_pages
|
|
|
|
self.log('copying snippets')
|
|
self.copy_snippets
|
|
|
|
self.log('copying content types')
|
|
self.copy_content_types
|
|
|
|
self.log('copying config files')
|
|
self.copy_config_files
|
|
|
|
self.log('generating the zip file')
|
|
self.zip_it!
|
|
end
|
|
|
|
# returns the path to the zipfile
|
|
def self.run!(site, filename = nil)
|
|
@@instance = self.new(site, filename)
|
|
@@instance.run!
|
|
end
|
|
|
|
protected
|
|
|
|
def zip_it!
|
|
"#{@target_folder}.zip".tap do |dst|
|
|
FileUtils.rm(dst, :force => true)
|
|
::Zip::ZipFile.open(dst, ::Zip::ZipFile::CREATE) do |zipfile|
|
|
Dir[File.join(@target_folder, '**/*')].each do |file|
|
|
entry = file.gsub(@target_folder + '/', '')
|
|
zipfile.add(entry, file)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
def create_target_folder
|
|
FileUtils.rm_rf(@target_folder)
|
|
FileUtils.mkdir_p(@target_folder)
|
|
%w(app/content_types app/views/snippets app/views/pages config data public).each do |f|
|
|
FileUtils.mkdir_p(File.join(@target_folder, f))
|
|
end
|
|
end
|
|
|
|
def initialize_site_hash
|
|
attributes = self.extract_attributes(@site, %w(name locale seo_title meta_description meta_keywords))
|
|
|
|
attributes['pages'] = []
|
|
attributes['content_types'] = {}
|
|
|
|
@site_hash = { 'site' => attributes }
|
|
end
|
|
|
|
def copy_config_files
|
|
File.open(File.join(self.config_folder, 'compiled_site.yml'), 'w') do |f|
|
|
f.write(yaml(@site_hash))
|
|
end
|
|
|
|
@site_hash['site'].delete('content_types')
|
|
|
|
File.open(File.join(self.config_folder, 'site.yml'), 'w') do |f|
|
|
f.write(yaml(@site_hash))
|
|
end
|
|
end
|
|
|
|
def copy_pages
|
|
Page.quick_tree(@site, false).each { |p| self._copy_pages(p) }
|
|
end
|
|
|
|
def copy_snippets
|
|
@site.snippets.each do |snippet|
|
|
File.open(File.join(self.snippets_folder, "#{snippet.slug}.liquid"), 'w') do |f|
|
|
f.write(snippet.template)
|
|
end
|
|
end
|
|
end
|
|
|
|
def copy_content_types
|
|
@site.content_types.each do |content_type|
|
|
attributes = self.extract_attributes(content_type, %w(name description slug order_by order_direction group_by_field_name api_enabled))
|
|
|
|
attributes['highlighted_field_name'] = content_type.highlighted_field._alias
|
|
|
|
# custom_fields
|
|
fields = []
|
|
content_type.contents_custom_fields.each do |field|
|
|
field_attributes = self.extract_attributes(field, %w(label kind hint required))
|
|
|
|
if field.target.present?
|
|
target_klass = field['target'].constantize
|
|
|
|
field_attributes['target'] = target_klass._parent.slug
|
|
|
|
if field['reverse_lookup'].present?
|
|
field_attributes['reverse'] = target_klass.custom_field_name_to_alias(field['reverse_lookup'])
|
|
end
|
|
end
|
|
|
|
fields << { field._alias => field_attributes }
|
|
end
|
|
|
|
attributes['fields'] = fields
|
|
|
|
@site_hash['site']['content_types'][content_type.name] = attributes.clone
|
|
|
|
# [structure] copy it into its own file
|
|
File.open(File.join(self.content_types_folder, "#{content_type.slug}.yml"), 'w') do |f|
|
|
f.write(self.yaml(attributes))
|
|
end
|
|
|
|
# data
|
|
data = self.extract_contents(content_type)
|
|
|
|
# [data] copy them into their own file
|
|
File.open(File.join(self.content_data_folder, "#{content_type.slug}.yml"), 'w') do |f|
|
|
f.write(self.yaml(data))
|
|
end
|
|
|
|
@site_hash['site']['content_types'][content_type.name]['contents'] = data
|
|
end
|
|
end
|
|
|
|
def copy_theme_assets
|
|
@site.theme_assets.each do |theme_asset|
|
|
target_path = File.join(self.public_folder, theme_asset.local_path)
|
|
|
|
self.copy_file_from_an_uploader(theme_asset.source, target_path) do |bytes|
|
|
if theme_asset.stylesheet_or_javascript?
|
|
base_url = theme_asset.source.url.gsub(theme_asset.local_path, '')
|
|
bytes.gsub(base_url, '/')
|
|
else
|
|
bytes
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
def copy_content_assets
|
|
@site.content_assets.each do |content_asset|
|
|
target_path = File.join(self.samples_folder, content_asset.source_filename)
|
|
self.copy_file_from_an_uploader(content_asset.source, target_path)
|
|
end
|
|
end
|
|
|
|
protected
|
|
|
|
def _copy_pages(page)
|
|
attributes = self.extract_attributes(page, %w{title seo_title meta_description meta_keywords redirect_url content_type published})
|
|
|
|
attributes['listed'] = page.listed? # in some cases, page.listed can be nil
|
|
|
|
unless page.raw_template.blank?
|
|
attributes.delete('redirect_url')
|
|
|
|
if page.templatized?
|
|
attributes['content_type'] = page.content_type.slug
|
|
end
|
|
|
|
# add editable elements
|
|
page.editable_elements.each do |element|
|
|
next if element.disabled? || (element.from_parent && element.default_content == element.content)
|
|
|
|
el_attributes = self.extract_attributes(element, %w{slug block hint})
|
|
|
|
case element
|
|
when EditableShortText, EditableLongText
|
|
el_attributes['content'] = self.replace_asset_urls_in(element.content || '')
|
|
when EditableFile
|
|
unless element.source_filename.blank?
|
|
filepath = File.join('/', 'samples', element.source_filename)
|
|
self.copy_file_from_an_uploader(element.source, File.join(self.public_folder, filepath))
|
|
el_attributes['content'] = filepath
|
|
else
|
|
el_attributes['content'] = '' # not sure if we run into this
|
|
end
|
|
end
|
|
|
|
(attributes['editable_elements'] ||= []) << el_attributes
|
|
end
|
|
|
|
page_templatepath = File.join(self.pages_folder, "#{page.fullpath}.liquid")
|
|
|
|
FileUtils.mkdir_p(File.dirname(page_templatepath))
|
|
|
|
File.open(page_templatepath, 'w') do |f|
|
|
f.write(self.replace_asset_urls_in(page.raw_template))
|
|
end
|
|
end
|
|
|
|
@site_hash['site']['pages'] << { page.fullpath => attributes }
|
|
|
|
page.children.each { |p| self._copy_pages(p) }
|
|
end
|
|
|
|
def extract_attributes(object, fields)
|
|
attributes = object.attributes.select { |k, v| fields.include?(k) && !v.blank? }
|
|
|
|
if RUBY_VERSION =~ /1\.9/
|
|
attributes
|
|
else
|
|
attributes.inject({}) { |memo, pair| memo.merge(pair.first => pair.last) }
|
|
end
|
|
end
|
|
|
|
def pages_folder
|
|
File.join(@target_folder, 'app', 'views', 'pages')
|
|
end
|
|
|
|
def snippets_folder
|
|
File.join(@target_folder, 'app', 'views', 'snippets')
|
|
end
|
|
|
|
def content_types_folder
|
|
File.join(@target_folder, 'app', 'content_types')
|
|
end
|
|
|
|
def config_folder
|
|
File.join(@target_folder, 'config')
|
|
end
|
|
|
|
def content_data_folder
|
|
File.join(@target_folder, 'data')
|
|
end
|
|
|
|
def public_folder
|
|
File.join(@target_folder, 'public')
|
|
end
|
|
|
|
def samples_folder
|
|
File.join(@target_folder, 'public', 'samples')
|
|
end
|
|
|
|
def copy_file_from_an_uploader(uploader, target_path, &block)
|
|
FileUtils.mkdir_p(File.dirname(target_path))
|
|
File.open(target_path, 'w') do |f|
|
|
bytes = uploader.read
|
|
|
|
bytes ||= ''
|
|
|
|
bytes = block.call(bytes) if block_given?
|
|
|
|
bytes = bytes.force_encoding('UTF-8') if RUBY_VERSION =~ /1\.9/
|
|
f.write(bytes)
|
|
end
|
|
end
|
|
|
|
def extract_contents(content_type)
|
|
data = []
|
|
|
|
highlighted_field_name = content_type.highlighted_field_name
|
|
|
|
content_type.contents.each do |content|
|
|
hash = {}
|
|
|
|
content.custom_fields.each do |field|
|
|
next if field._name == highlighted_field_name
|
|
|
|
value = (case field.kind
|
|
when 'file'
|
|
uploader = content.send(field._name)
|
|
unless uploader.blank?
|
|
filepath = File.join('/samples', content_type.slug, content.send("#{field._name}_filename"))
|
|
self.copy_file_from_an_uploader(uploader, File.join(self.public_folder, filepath))
|
|
else
|
|
filepath = nil
|
|
end
|
|
filepath
|
|
when 'text'
|
|
self.replace_asset_urls_in(content.send(field._name.to_sym) || '')
|
|
when 'has_one'
|
|
content.send(field.safe_alias.to_sym).highlighted_field_value rescue nil # no bound object
|
|
when 'has_many'
|
|
unless field.reverse_has_many?
|
|
content.send(field.safe_alias.to_sym).collect(&:highlighted_field_value)
|
|
end
|
|
else
|
|
content.send(field.safe_alias.to_sym)
|
|
end)
|
|
|
|
hash[field._alias] = value unless value.blank?
|
|
end
|
|
|
|
data << { content.highlighted_field_value => hash }
|
|
end
|
|
|
|
data
|
|
end
|
|
|
|
def replace_asset_urls_in(text)
|
|
base_url = ContentAssetUploader.new(ContentAsset.new(:site => @site)).store_dir
|
|
(base_url = base_url.split('/')).pop
|
|
base_url = base_url.join('/')
|
|
|
|
text.gsub(%r(#{base_url}/[a-z0-9]{10,}/), "#{base_url.starts_with?('http') ? '/' : ''}samples/")
|
|
end
|
|
|
|
def yaml(hash_or_array)
|
|
method = hash_or_array.respond_to?(:ya2yaml) ? :ya2yaml : :to_yaml
|
|
string = (if hash_or_array.respond_to?(:keys)
|
|
hash_or_array.dup.stringify_keys!
|
|
else
|
|
hash_or_array
|
|
end).send(method)
|
|
string.gsub('!ruby/symbol ', ':').sub('---', '').split("\n").map(&:rstrip).join("\n").strip
|
|
end
|
|
|
|
def log(message, domain = '')
|
|
head = "[export_template][#{@site.name}]"
|
|
head += "[#{domain}]" unless domain.blank?
|
|
::Locomotive.log "\t#{head} #{message}"
|
|
end
|
|
|
|
end
|
|
|
|
end
|