applicationsonline/librarian

View on GitHub
lib/librarian/lockfile/parser.rb

Summary

Maintainability
A
1 hr
Test Coverage
require 'librarian/manifest'
require 'librarian/dependency'
require 'librarian/manifest_set'

module Librarian
  class Lockfile
    class Parser

      class ManifestPlaceholder
        attr_reader :source, :name, :version, :dependencies
        def initialize(source, name, version, dependencies)
          @source, @name, @version, @dependencies = source, name, version, dependencies
        end
      end

      attr_accessor :environment
      private :environment=

      def initialize(environment)
        self.environment = environment
      end

      def parse(string)
        lines = string.lines.map{|l| l.sub(/\s+\z/, '')}.reject(&:empty?)
        sources = extract_and_parse_sources(lines)
        manifests = compile(sources)
        manifests_index = Hash[manifests.map{|m| [m.name, m]}]
        raise StandardError, "Expected DEPENDENCIES topic!" unless lines.shift == "DEPENDENCIES"
        dependencies = extract_and_parse_dependencies(lines, manifests_index)
        Resolution.new(dependencies, manifests)
      end

    private

      def extract_and_parse_sources(lines)
        sources = []
        while source_type_names.include?(lines.first)
          source = {}
          source_type_name = lines.shift
          source[:type] = source_type_names_map[source_type_name]
          options = {}
          while lines.first =~ /^ {2}([\w-]+):\s+(.+)$/
            lines.shift
            options[$1.to_sym] = $2
          end
          source[:options] = options
          lines.shift # specs
          manifests = {}
          while lines.first =~ /^ {4}([\w-]+) \((.*)\)$/
            lines.shift
            name = $1
            manifests[name] = {:version => $2, :dependencies => {}}
            while lines.first =~ /^ {6}([\w-]+) \((.*)\)$/
              lines.shift
              manifests[name][:dependencies][$1] = $2.split(/,\s*/)
            end
          end
          source[:manifests] = manifests
          sources << source
        end
        sources
      end

      def extract_and_parse_dependencies(lines, manifests_index)
        dependencies = []
        while lines.first =~ /^ {2}([\w-]+)(?: \((.*)\))?$/
          lines.shift
          name, requirement = $1, $2.split(/,\s*/)
          dependencies << Dependency.new(name, requirement, manifests_index[name].source)
        end
        dependencies
      end

      def compile_placeholder_manifests(sources_ast)
        manifests = {}
        sources_ast.each do |source_ast|
          source_type = source_ast[:type]
          source = source_type.from_lock_options(environment, source_ast[:options])
          source_ast[:manifests].each do |manifest_name, manifest_ast|
            manifests[manifest_name] = ManifestPlaceholder.new(
              source,
              manifest_name,
              manifest_ast[:version],
              manifest_ast[:dependencies].map{|k, v| Dependency.new(k, v, nil)}
            )
          end
        end
        manifests
      end

      def compile(sources_ast)
        manifests = compile_placeholder_manifests(sources_ast)
        manifests = manifests.map do |name, manifest|
          dependencies = manifest.dependencies.map do |d|
            Dependency.new(d.name, d.requirement, manifests[d.name].source)
          end
          real = Manifest.new(manifest.source, manifest.name)
          real.version = manifest.version
          real.dependencies = manifest.dependencies
          real
        end
        ManifestSet.sort(manifests)
      end

      def dsl_class
        environment.dsl_class
      end

      def source_type_names_map
        @source_type_names_map ||= begin
          Hash[dsl_class.source_types.map{|t| [t[1].lock_name, t[1]]}]
        end
      end

      def source_type_names
        @source_type_names ||= begin
          dsl_class.source_types.map{|t| t[1].lock_name}
        end
      end

    end
  end
end