require 'minimart/utils/archive' require 'minimart/utils/http' module Minimart module Web # This class is responsible for generating the universe.json file, and the # related directory structure of gzipped cookbooks. class UniverseGenerator include TemplateHelper # @return [String] the directory to put the universe.json file in attr_reader :web_directory # @return [String] the base URL to use to build paths for cookbook files. attr_reader :endpoint # @return [Minimart::Web::Cookbooks] The cookbooks to build a universe for attr_reader :cookbooks attr_reader :universe # @param [Hash] opts # @option opts [String] web_directory The directory to put the universe.json file in # @option opts [String] endpoint The base URL to use to build paths for cookbook files. # @option opts [Minimart::Web::Cookbooks] cookbooks The cookbooks to build a universe for # @option opts [Boolean] clean_cookbooks Determines if we should clean the cookbook_files folder and recreate cookbook packages def initialize(opts = {}) @web_directory = opts[:web_directory] @endpoint = opts[:endpoint] @cookbooks = opts[:cookbooks] @clean_cookbooks = opts.fetch(:clean_cookbooks, true) @universe = {} end # Generate the universe file! def generate clean_existing_cookbook_files make_cookbook_files_directory create_universe write_universe_file end private def clean_existing_cookbook_files if Dir.exists?(cookbook_files_directory) && @clean_cookbooks FileUtils.remove_entry(cookbook_files_directory) end end def make_cookbook_files_directory FileUtils.mkdir_p(cookbook_files_directory) end def create_universe cookbooks.individual_cookbooks.each do |cookbook| unless Dir.exists?(archive_directory(cookbook)) make_cookbook_directory(cookbook) generate_archive_file(cookbook) end add_cookbook_to_universe(cookbook) end end def make_cookbook_directory(cookbook) FileUtils.mkdir_p(cookbook_directory(cookbook.name)) end # /web/cookbook_files/cookbook-name def cookbook_directory(cookbook_name) File.join(cookbook_files_directory, cookbook_name) end def generate_archive_file(cookbook) FileUtils.mkdir_p(archive_directory(cookbook)) Utils::Archive.pack_archive(cookbook, archive_name(cookbook)) end def archive_name(cookbook) File.join(archive_directory(cookbook), "#{cookbook}.tar.gz") end # /web/cookbook_files/cookbook-name/cookbook-version def archive_directory(cookbook) File.join(cookbook_directory(cookbook.name), cookbook.web_friendly_version) end def add_cookbook_to_universe(cookbook) universe[cookbook.name] ||= {} universe[cookbook.name][cookbook.version.to_s] = { location_type: :uri, location_path: archive_url(cookbook), download_url: archive_url(cookbook), dependencies: cookbook.dependencies } end def archive_url(cookbook) Utils::Http.build_url(endpoint, cookbook_download_path(cookbook)) end def write_universe_file File.open(File.join(web_directory, 'universe'), 'w+') do |f| f.write(universe.to_json) end end def cookbook_files_directory @cookbook_files_directory ||= File.join(web_directory, '/cookbook_files') end end end end