CocoaPods/cocoapods-try

View on GitHub
lib/pod/command/try.rb

Summary

Maintainability
A
1 hr
Test Coverage
require 'pod/try_settings'

# The CocoaPods namespace
#
module Pod
  class Command
    # The pod try command.
    # @CocoaPods 0.29.0
    #
    class Try < Command
      include RepoUpdate

      self.summary = 'Try a Pod!'

      self.description = <<-DESC
          Downloads the Pod with the given `NAME` (or Git `URL`), install its
          dependencies if needed and opens its demo project. If a Git URL is
          provided the head of the repo is used.

          If a Git URL is specified, then a --podspec_name can be provided,
          if the podspec name is different than the git repo for some reason.
      DESC

      self.arguments = [CLAide::Argument.new(%w(NAME URL), true)]

      def self.options
        [
          ['--podspec_name=[name]', 'The name of the podspec file within the Git Repository'],
        ].concat(super)
      end

      def initialize(argv)
        @name = argv.shift_argument
        @podspec_name = argv.option('podspec_name')
        super
      end

      def validate!
        super
        help! 'A Pod name or URL is required.' unless @name
        help! 'Podspec name can only be used with a Git URL' if @podspec_name && !git_url?(@name)
      end

      def run
        ensure_master_spec_repo_exists!
        sandbox = Sandbox.new(TRY_TMP_DIR)
        spec = setup_spec_in_sandbox(sandbox)

        UI.title "Trying #{spec.name}" do
          pod_dir = install_pod(spec, sandbox)
          settings = TrySettings.settings_from_folder(pod_dir)
          Dir.chdir(pod_dir) { settings.run_pre_install_commands(true) }
          proj = settings.project_path || pick_demo_project(pod_dir)
          file = install_podfile(proj)
          if file
            open_project(file)
          else
            UI.puts "Unable to locate a project for #{spec.name}"
          end
        end
      end

      public

      # Helpers
      #-----------------------------------------------------------------------#

      # @return [Pathname]
      #
      TRY_TMP_DIR = Pathname.new(Dir.tmpdir) + 'CocoaPods/Try'

      # Puts the spec's data in the sandbox
      #
      def setup_spec_in_sandbox(sandbox)
        if git_url?(@name)
          spec = spec_with_url(@name, @podspec_name)
          sandbox.store_pre_downloaded_pod(spec.name)
        else
          update_specs_repos
          spec = spec_with_name(@name)
        end
        spec
      end

      # Returns the specification of the last version of the Pod with the given
      # name.
      #
      # @param  [String] name
      #         The name of the pod.
      #
      # @return [Specification] The specification.
      #
      def spec_with_name(name)
        set = config.sources_manager.search(Dependency.new(name))
        if set
          set.specification.root
        else
          raise Informative, "Unable to find a specification for `#{name}`"
        end
      end

      # Returns the specification found in the given Git repository URL by
      # downloading the repository.
      #
      # @param [String] url
      #        The URL for the pod Git repository.
      #
      # @param [String] spec_name
      #        The name of the podspec file within the Git repository.
      #
      # @return [Specification] The specification.
      #
      def spec_with_url(url, spec_name = nil)
        name = url.split('/').last
        name = name.chomp('.git') if name.end_with?('.git')
        name = spec_name unless spec_name.nil?

        target_dir = TRY_TMP_DIR + name
        target_dir.rmtree if target_dir.exist?

        downloader = Pod::Downloader.for_target(target_dir, :git => url)
        downloader.download

        spec_file = Pathname.glob(target_dir + "#{name}.podspec{,.json}").first
        Pod::Specification.from_file(spec_file)
      end

      # Installs the specification in the given directory.
      #
      # @param  [Specification] The specification of the Pod.
      # @param  [Pathname] The directory of the sandbox where to install the
      #         Pod.
      #
      # @return [Pathname] The path where the Pod was installed
      #
      def install_pod(spec, sandbox)
        specs = { :ios => spec, :osx => spec }
        if cocoapods_version >= Pod::Version.new('1.8.0')
          dummy_podfile = Podfile.new
          installer = Installer::PodSourceInstaller.new(sandbox, dummy_podfile, specs, :can_cache => false)
        else
          installer = Installer::PodSourceInstaller.new(sandbox, specs, :can_cache => false)
        end
        installer.install!
        sandbox.root + spec.name
      end

      # Picks a project or workspace suitable for the demo purposes in the
      # given directory.
      #
      # To decide the project simple heuristics are used according to the name,
      # if no project is found this method raises and `Informative` otherwise
      # if more than one project is found the choice is presented to the user.
      #
      # @param  [#to_s] dir
      #         The path where to look for projects.
      #
      # @return [String] The path of the project.
      #
      def pick_demo_project(dir)
        dir = Pathname.new(dir)
        projs = projects_in_dir(dir)
        if projs.count == 0
          raise Informative, 'Unable to find any project in the source files' \
            " of the Pod: `#{dir}`"
        elsif projs.count == 1
          projs.first
        elsif (workspaces = projs.grep(/(demo|example|sample).*\.xcworkspace$/i)).count == 1
          workspaces.first
        elsif (projects = projs.grep(/demo|example|sample/i)).count == 1
          projects.first
        else
          message = 'Which project would you like to open'
          selection_array = projs.map do |p|
            Pathname.new(p).relative_path_from(dir).to_s
          end
          index = UI.choose_from_array(selection_array, message)
          projs[index]
        end
      end

      # Performs a CocoaPods installation for the given project if Podfile is
      # found.  Shells out to avoid issues with the config of the process
      # running the try command.
      #
      # @return [String] proj
      #         The path of the project.
      #
      # @return [String] The path of the file to open, in other words the
      #         workspace of the installation or the given project.
      #
      def install_podfile(proj)
        return unless proj
        dirname = Pathname.new(proj).dirname
        podfile_path = dirname + 'Podfile'
        if podfile_path.exist?
          Dir.chdir(dirname) do
            perform_cocoapods_installation

            podfile = Pod::Podfile.from_file(podfile_path)

            if podfile.workspace_path
              File.expand_path(podfile.workspace_path)
            else
              proj.to_s.chomp(File.extname(proj.to_s)) + '.xcworkspace'
            end
          end
        else
          proj
        end
      end

      public

      # Private Helpers
      #-----------------------------------------------------------------------#

      # @return [void] Updates the specs repo unless disabled by the config.
      #
      def update_specs_repos
        return unless repo_update?(:default => true)
        UI.section 'Updating spec repositories' do
          config.sources_manager.update
        end
      end

      # Opens the project at the given path.
      #
      # @return [String] path
      #         The path of the project.
      #
      # @return [void]
      #
      def open_project(path)
        UI.puts "Opening '#{path}'"
        `open "#{path}"`
      end

      # @return [void] Performs a CocoaPods installation in the working
      #         directory.
      #
      def perform_cocoapods_installation
        UI.titled_section 'Performing CocoaPods Installation' do
          Command::Install.invoke
        end
      end

      # @return [Bool] Wether the given string is the name of a Pod or an URL
      #         for a Git repo.
      #
      def git_url?(name)
        prefixes = ['https://', 'http://']
        prefixes.any? { |prefix| name.start_with?(prefix) }
      end

      # @return [Array<String>] The list of the workspaces and projects in a
      #         given directory excluding The Pods project and the projects
      #         that have a sister workspace.
      #
      def projects_in_dir(dir)
        glob_match = Dir.glob("#{dir}/**/*.xc{odeproj,workspace}")
        glob_match = glob_match.reject do |p|
          next true if p.include?('Pods.xcodeproj')
          next true if p.end_with?('.xcodeproj/project.xcworkspace')
          sister_workspace = p.chomp(File.extname(p.to_s)) + '.xcworkspace'
          p.end_with?('.xcodeproj') && glob_match.include?(sister_workspace)
        end
      end

      # @return [Pod::Version] the version of CocoaPods currently running
      #
      def cocoapods_version
        Pod::Version.new(Pod::VERSION)
      end

      #-------------------------------------------------------------------#
    end
  end
end