File: parser.rb

package info (click to toggle)
ruby-librarian 1.2.0-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 632 kB
  • sloc: ruby: 6,109; makefile: 11
file content (123 lines) | stat: -rw-r--r-- 3,989 bytes parent folder | download | duplicates (6)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
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 << environment.dsl_class.dependency_type.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| environment.dsl_class.dependency_type.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|
            environment.dsl_class.dependency_type.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