babl-json 0.5.9 → 0.6.0

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA1:
3
- metadata.gz: 329322cf3fa5b7e83cad065409efcd8889146f0f
4
- data.tar.gz: 4b492ee8bc38517fcd1c2e138631e95301ea655f
3
+ metadata.gz: 1b50e6e799274d6d0286709ee0484a75dea2e09f
4
+ data.tar.gz: 5b728f4581bea01d71aaf62af160f06b1c0551bc
5
5
  SHA512:
6
- metadata.gz: 3a0b504aa4bfdd699fe1c7bb5d2e42096329d503fe439878a13c9bbe6239f30de74a3c68fbb4d1d2309ec6f3b87a254117489f8d6d874c4d83e8b0daba00952a
7
- data.tar.gz: ae23a4755c56670770efdcc4dfa1abf1fed22ef6b48ce954cf2d726908c9652d768571ee8a735eee79606ab0c8ad83c6991a90e753b2ff73fc427f3de6ef2539
6
+ metadata.gz: b0af0e4fa77e6971d03230617cdf13c211d6587603bd509c531faf9d9e9358b1e88eb982be6df48551ac8033ed190f7d52357988e987ee3bc5c57f609d8528a1
7
+ data.tar.gz: c88f3b618e4fe2f2dc9181ed07ee6333d7cdfbcd5fff68071cba0a87b9155f9f1e7a13d89cb36cba68b8421773b4330767d1069e290f7522e171b4a9bd9ad532
@@ -52,6 +52,7 @@ module Babl
52
52
  query = File.join(search_path, "{#{partial_name}}{.babl,}")
53
53
  path = Dir[query].first
54
54
  return unless path
55
+
55
56
  source = File.read(path)
56
57
  [current_template.source(source, path, 0), self]
57
58
  end
@@ -76,6 +77,7 @@ module Babl
76
77
  def template
77
78
  cached = @cached_template
78
79
  return cached.last if cached && [config.using].flatten == cached.first
80
+
79
81
  # Calling 'using' is a very inefficient operation, because
80
82
  # it creates a new class. We can avoid that cost most of the
81
83
  # time, assuming 'config.using' does not change often (typically
@@ -29,8 +29,8 @@ module Babl
29
29
  @scope = block
30
30
  end
31
31
 
32
- def precompile(node, **context)
33
- bind(BoundOperator.new(context)).precompile(node)
32
+ def precompile(node, context)
33
+ bind(BoundOperator.new(context, &:itself)).scope.call(node)
34
34
  end
35
35
 
36
36
  def bind(bound)
@@ -43,38 +43,38 @@ module Babl
43
43
  unless [Nodes::InternalValue.instance, Nodes::TerminalValue.instance].include?(node)
44
44
  raise Errors::InvalidTemplate, 'Chaining is not allowed after a terminal operator'
45
45
  end
46
+
46
47
  yield context
47
48
  end
48
49
  end
49
50
 
50
51
  # Append an operator to the chain, and return a new Builder object
51
- def construct_node(**new_context)
52
+ def construct_node
53
+ wrap { |bound|
54
+ BoundOperator.new(bound.context) { |node| bound.scope[yield(node, bound.context)] }
55
+ }
56
+ end
57
+
58
+ def construct_context
52
59
  wrap { |bound|
53
- bound.nest(bound.context.merge(new_context)) { |node|
54
- yield(node, bound.context)
55
- }
60
+ new_context = yield(bound.context)
61
+ next bound if bound.context.equal?(new_context)
62
+
63
+ BoundOperator.new(new_context, &bound.scope)
56
64
  }
57
65
  end
58
66
 
59
67
  def wrap
60
- self.class.new { |bound| yield bind(bound) }
68
+ ChainBuilder.new { |bound| yield bind(bound) }
61
69
  end
62
70
  end
63
71
 
64
72
  class BoundOperator
65
- attr_reader :context
73
+ attr_reader :context, :scope
66
74
 
67
75
  def initialize(context, &scope)
68
76
  @context = context
69
- @scope = scope || :itself.to_proc
70
- end
71
-
72
- def precompile(node)
73
- @scope[node]
74
- end
75
-
76
- def nest(context)
77
- self.class.new(context) { |node| @scope[yield node] }
77
+ @scope = scope
78
78
  end
79
79
  end
80
80
  end
@@ -17,7 +17,7 @@ module Babl
17
17
  end
18
18
 
19
19
  def compile(preloader: Rendering::NoopPreloader, pretty: true, optimize: true, lookup_context: nil)
20
- tree = precompile(lookup_context: lookup_context)
20
+ tree = precompile(Nodes::TerminalValue.instance, lookup_context: lookup_context)
21
21
  tree = tree.optimize if optimize
22
22
  validate(tree)
23
23
 
@@ -46,12 +46,16 @@ module Babl
46
46
  # NOOP
47
47
  end
48
48
 
49
- def precompile(node = Nodes::TerminalValue.instance, **context)
50
- builder.precompile(node, **context)
49
+ def precompile(node, context)
50
+ builder.precompile(node, context)
51
51
  end
52
52
 
53
- def construct_node(**new_context, &block)
54
- self.class.new builder.construct_node(**new_context, &block)
53
+ def construct_node(&block)
54
+ self.class.new builder.construct_node(&block)
55
+ end
56
+
57
+ def construct_context(&block)
58
+ self.class.new builder.construct_context(&block)
55
59
  end
56
60
 
57
61
  def construct_terminal(&block)
@@ -7,11 +7,11 @@ module Babl
7
7
  module Nodes
8
8
  class Concat < Utils::Value.new(:nodes)
9
9
  memoize def dependencies
10
- Babl::Utils::Hash.deep_merge(*nodes.map(&:dependencies))
10
+ Babl::Utils::Hash.deep_merge(nodes.map(&:dependencies))
11
11
  end
12
12
 
13
13
  memoize def pinned_dependencies
14
- Babl::Utils::Hash.deep_merge(*nodes.map(&:pinned_dependencies))
14
+ Babl::Utils::Hash.deep_merge(nodes.map(&:pinned_dependencies))
15
15
  end
16
16
 
17
17
  memoize def schema
@@ -47,6 +47,7 @@ module Babl
47
47
 
48
48
  def optimize_single
49
49
  return unless nodes.size == 1
50
+
50
51
  optimized = nodes.first.optimize
51
52
 
52
53
  case optimized
@@ -62,6 +63,7 @@ module Babl
62
63
  def optimize_concatenated_arrays
63
64
  optimized_nodes = nodes.map(&:optimize)
64
65
  return if optimized_nodes == nodes
66
+
65
67
  Concat.new(optimized_nodes).optimize
66
68
  end
67
69
 
@@ -70,6 +72,7 @@ module Babl
70
72
  obj1 = constant_to_array(obj1) if Constant === obj1
71
73
  obj2 = constant_to_array(obj2) if Constant === obj2
72
74
  next unless FixedArray === obj1 && FixedArray === obj2
75
+
73
76
  new_nodes = nodes.dup
74
77
  new_nodes[idx] = FixedArray.new(obj1.nodes + obj2.nodes)
75
78
  new_nodes[idx + 1] = nil
@@ -15,7 +15,7 @@ module Babl
15
15
  end
16
16
 
17
17
  memoize def dependencies
18
- Babl::Utils::Hash.deep_merge(node.dependencies, node.pinned_dependencies[ref] || Utils::Hash::EMPTY)
18
+ Babl::Utils::Hash.deep_merge([node.dependencies, node.pinned_dependencies[ref] || Utils::Hash::EMPTY])
19
19
  end
20
20
 
21
21
  memoize def pinned_dependencies
@@ -17,7 +17,7 @@ module Babl
17
17
  end
18
18
 
19
19
  memoize def dependencies
20
- Babl::Utils::Hash.deep_merge(node.dependencies, path)
20
+ Babl::Utils::Hash.deep_merge([node.dependencies, path])
21
21
  end
22
22
 
23
23
  memoize def optimize
@@ -10,10 +10,10 @@ module Babl
10
10
  node_deps = node.dependencies
11
11
  child_deps = node_deps.reject { |key, _| key == Parent::PARENT_MARKER }
12
12
 
13
- Babl::Utils::Hash.deep_merge(
13
+ Babl::Utils::Hash.deep_merge([
14
14
  node_deps[Parent::PARENT_MARKER] || Utils::Hash::EMPTY,
15
15
  __each__: child_deps
16
- )
16
+ ])
17
17
  end
18
18
 
19
19
  memoize def schema
@@ -13,11 +13,11 @@ module Babl
13
13
  end
14
14
 
15
15
  memoize def dependencies
16
- Babl::Utils::Hash.deep_merge(*nodes.map(&:dependencies))
16
+ Babl::Utils::Hash.deep_merge(nodes.map(&:dependencies))
17
17
  end
18
18
 
19
19
  memoize def pinned_dependencies
20
- Babl::Utils::Hash.deep_merge(*nodes.map(&:pinned_dependencies))
20
+ Babl::Utils::Hash.deep_merge(nodes.map(&:pinned_dependencies))
21
21
  end
22
22
 
23
23
  memoize def optimize
@@ -10,7 +10,7 @@ module Babl
10
10
  end
11
11
 
12
12
  memoize def pinned_dependencies
13
- Babl::Utils::Hash.deep_merge(node.pinned_dependencies, ref => node.dependencies)
13
+ Babl::Utils::Hash.deep_merge([node.pinned_dependencies, ref => node.dependencies])
14
14
  end
15
15
 
16
16
  memoize def schema
@@ -8,11 +8,11 @@ module Babl
8
8
  module Nodes
9
9
  class Merge < Utils::Value.new(:nodes)
10
10
  memoize def dependencies
11
- Babl::Utils::Hash.deep_merge(*nodes.map(&:dependencies))
11
+ Babl::Utils::Hash.deep_merge(nodes.map(&:dependencies))
12
12
  end
13
13
 
14
14
  memoize def pinned_dependencies
15
- Babl::Utils::Hash.deep_merge(*nodes.map(&:pinned_dependencies))
15
+ Babl::Utils::Hash.deep_merge(nodes.map(&:pinned_dependencies))
16
16
  end
17
17
 
18
18
  memoize def schema
@@ -44,6 +44,7 @@ module Babl
44
44
 
45
45
  def optimize_single
46
46
  return unless nodes.size == 1
47
+
47
48
  optimized = nodes.first.optimize
48
49
 
49
50
  case optimized
@@ -59,11 +60,13 @@ module Babl
59
60
  def optimize_merged_objects
60
61
  optimized_nodes = nodes.map(&:optimize)
61
62
  return if optimized_nodes == nodes
63
+
62
64
  Merge.new(optimized_nodes).optimize
63
65
  end
64
66
 
65
67
  def optimize_nested_merges
66
68
  return unless nodes.any? { |node| Merge === node }
69
+
67
70
  Merge.new(nodes.flat_map { |node| Merge === node ? node.nodes : [node] }).optimize
68
71
  end
69
72
 
@@ -73,6 +76,7 @@ module Babl
73
76
  obj2 = constant_to_object(obj2) if Constant === obj2
74
77
 
75
78
  next unless Object === obj1 && Object === obj2
79
+
76
80
  new_nodes = nodes.dup
77
81
  new_nodes[idx] = Object.new(obj1.nodes.merge(obj2.nodes))
78
82
  new_nodes[idx + 1] = nil
@@ -10,10 +10,10 @@ module Babl
10
10
  node_deps = node.dependencies
11
11
  child_deps = node_deps.reject { |key, _| key == Parent::PARENT_MARKER }
12
12
 
13
- Babl::Utils::Hash.deep_merge(
13
+ Babl::Utils::Hash.deep_merge([
14
14
  node_deps[Parent::PARENT_MARKER] || Utils::Hash::EMPTY,
15
15
  property => child_deps
16
- )
16
+ ])
17
17
  end
18
18
 
19
19
  memoize def schema
@@ -29,16 +29,17 @@ module Babl
29
29
  return optimized if Constant === optimized || GotoPin === optimized
30
30
  return optimized.node if Parent === optimized
31
31
  return self if optimized.equal?(node)
32
+
32
33
  Nav.new(property, optimized)
33
34
  end
34
35
 
35
36
  def render(frame)
36
37
  value = begin
37
- object = frame.object
38
- ::Hash === object ? object.fetch(property) : object.send(property)
39
- rescue StandardError => e
40
- raise Errors::RenderingError, "#{e.message}\n" + frame.formatted_stack(property), e.backtrace
41
- end
38
+ object = frame.object
39
+ ::Hash === object ? object.fetch(property) : object.send(property)
40
+ rescue StandardError => e
41
+ raise Errors::RenderingError, "#{e.message}\n" + frame.formatted_stack(property), e.backtrace
42
+ end
42
43
 
43
44
  frame.move_forward(value, property) do |new_frame|
44
45
  node.render(new_frame)
@@ -9,11 +9,11 @@ module Babl
9
9
  EMPTY = new(Utils::Hash::EMPTY)
10
10
 
11
11
  memoize def dependencies
12
- Babl::Utils::Hash.deep_merge(*nodes.values.map(&:dependencies))
12
+ Babl::Utils::Hash.deep_merge(nodes.values.map(&:dependencies))
13
13
  end
14
14
 
15
15
  memoize def pinned_dependencies
16
- Babl::Utils::Hash.deep_merge(*nodes.values.map(&:pinned_dependencies))
16
+ Babl::Utils::Hash.deep_merge(nodes.values.map(&:pinned_dependencies))
17
17
  end
18
18
 
19
19
  memoize def schema
@@ -27,6 +27,7 @@ module Babl
27
27
  object ||= Object.new(optimized_nodes)
28
28
 
29
29
  return object unless optimized_nodes.values.all? { |node| Constant === node }
30
+
30
31
  Constant.new(optimized_nodes.map { |k, v| [k, v.value] }.to_h.freeze, object.schema)
31
32
  end
32
33
 
@@ -23,6 +23,7 @@ module Babl
23
23
  optimized = node.optimize
24
24
  return optimized if Constant === optimized || GotoPin === optimized
25
25
  return self if optimized.equal?(node)
26
+
26
27
  Parent.new(optimized)
27
28
  end
28
29
 
@@ -10,15 +10,16 @@ module Babl
10
10
  class Switch < Utils::Value.new(:nodes)
11
11
  def initialize(nodes)
12
12
  raise Errors::InvalidTemplate, 'A least one switch() condition must be taken' if nodes.empty?
13
+
13
14
  super
14
15
  end
15
16
 
16
17
  memoize def dependencies
17
- Babl::Utils::Hash.deep_merge(*nodes.flatten(1).map(&:dependencies))
18
+ Babl::Utils::Hash.deep_merge(nodes.flatten(1).map(&:dependencies))
18
19
  end
19
20
 
20
21
  memoize def pinned_dependencies
21
- Babl::Utils::Hash.deep_merge(*nodes.flatten(1).map(&:pinned_dependencies))
22
+ Babl::Utils::Hash.deep_merge(nodes.flatten(1).map(&:pinned_dependencies))
22
23
  end
23
24
 
24
25
  memoize def schema
@@ -45,6 +46,7 @@ module Babl
45
46
  def optimize_continue_to_switch
46
47
  cond, val = nodes.last
47
48
  return unless Switch === val && Constant === cond && cond.value
49
+
48
50
  Switch.new(nodes[0...-1] + val.nodes).optimize
49
51
  end
50
52
 
@@ -52,6 +54,7 @@ module Babl
52
54
  conds = Set.new
53
55
  new_nodes = nodes.map { |cond, val|
54
56
  next if conds.include?(cond)
57
+
55
58
  conds << cond
56
59
  [cond, val]
57
60
  }.compact
@@ -61,6 +64,7 @@ module Babl
61
64
  def optimize_always_same_outputs
62
65
  return unless nodes.map(&:first).any? { |node| Constant === node && node.value }
63
66
  return unless nodes.map(&:last).uniq.size == 1
67
+
64
68
  nodes.first.last.optimize
65
69
  end
66
70
 
@@ -10,14 +10,14 @@ module Babl
10
10
  end
11
11
 
12
12
  memoize def dependencies
13
- Babl::Utils::Hash.deep_merge(
13
+ Babl::Utils::Hash.deep_merge([
14
14
  *nodes.map(&:dependencies),
15
15
  node.dependencies[Parent::PARENT_MARKER] || Utils::Hash::EMPTY
16
- )
16
+ ])
17
17
  end
18
18
 
19
19
  memoize def pinned_dependencies
20
- Babl::Utils::Hash.deep_merge(*(nodes + [node]).map(&:pinned_dependencies))
20
+ Babl::Utils::Hash.deep_merge((nodes + [node]).map(&:pinned_dependencies))
21
21
  end
22
22
 
23
23
  memoize def optimize
@@ -30,15 +30,16 @@ module Babl
30
30
 
31
31
  if optimized_nodes.all? { |n| Constant === n }
32
32
  value = begin
33
- block.call(*optimized_nodes.map(&:value))
34
- rescue StandardError => e
35
- raise Errors::InvalidTemplate, e.message, e.backtrace
36
- end
33
+ block.call(*optimized_nodes.map(&:value))
34
+ rescue StandardError => e
35
+ raise Errors::InvalidTemplate, e.message, e.backtrace
36
+ end
37
37
  constant_block = Utils::Proc.constant(value)
38
38
  return With.new(optimized, Utils::Array::EMPTY, constant_block)
39
39
  end
40
40
 
41
41
  return self if optimized.equal?(node) && optimized_nodes.each_with_index.all? { |n, idx| n.equal?(nodes[idx]) }
42
+
42
43
  With.new(optimized, optimized_nodes, block)
43
44
  end
44
45
 
@@ -7,11 +7,11 @@ module Babl
7
7
  module DSL
8
8
  # Produce an fixed-size array, using the provided templates to populate its elements.
9
9
  def array(*templates)
10
- templates = templates.map { |t| unscoped.call(t) }
10
+ templates = templates.map { |t| unscoped.reset_continue.call(t) }
11
11
 
12
12
  construct_terminal { |ctx|
13
13
  Nodes::FixedArray.new(templates.map { |t|
14
- t.builder.precompile(Nodes::TerminalValue.instance, ctx.merge(continue: nil))
14
+ t.builder.precompile(Nodes::TerminalValue.instance, ctx)
15
15
  })
16
16
  }
17
17
  end
@@ -24,11 +24,10 @@ module Babl
24
24
  case arg
25
25
  when Template then self.class.new(builder.wrap { |bound| arg.builder.bind(bound) })
26
26
  when Utils::DslProxy then call(arg.itself)
27
- when ::Symbol then nav(arg)
28
27
  when ::Proc then call(&arg)
29
28
  when ::Hash then object(arg)
30
29
  when ::Array then array(*arg)
31
- when ::String, ::Numeric, ::NilClass, ::TrueClass, ::FalseClass then static(arg)
30
+ when ::String, ::Numeric, ::NilClass, ::TrueClass, ::FalseClass, ::Symbol then static(arg)
32
31
  else raise Errors::InvalidTemplate, "call() received invalid argument: #{arg}"
33
32
  end
34
33
  end
@@ -8,14 +8,14 @@ module Babl
8
8
  # Produce an array by concatening the provided result of the given templates.
9
9
  # (they therefor have to produce arrays, or nil, which is interpreted as an empty array)
10
10
  def concat(*templates)
11
- templates = templates.map { |t| unscoped.call(t) }
11
+ templates = templates.map { |t| unscoped.reset_continue.call(t) }
12
12
 
13
13
  construct_terminal { |context|
14
14
  Nodes::Concat.new(
15
15
  templates.map { |t|
16
16
  t.builder.precompile(
17
17
  Nodes::TerminalValue.instance,
18
- context.merge(continue: nil)
18
+ context
19
19
  )
20
20
  }
21
21
  )
@@ -11,9 +11,21 @@ module Babl
11
11
  construct_terminal { |context|
12
12
  node = context[:continue]
13
13
  raise Errors::InvalidTemplate, 'continue() cannot be used outside switch()' unless node
14
+
14
15
  node
15
16
  }
16
17
  end
18
+
19
+ protected
20
+
21
+ # Clear contextual information about parent switch for the rest of the chain
22
+ def reset_continue
23
+ construct_context { |context|
24
+ next context unless context.key?(:continue)
25
+
26
+ context.reject { |k, _v| :continue == k }
27
+ }
28
+ end
17
29
  end
18
30
  end
19
31
  end
@@ -17,7 +17,7 @@ module Babl
17
17
  # but without navigating.
18
18
  def dep(*path)
19
19
  path = Dep.canonicalize(path)
20
- construct_node(continue: nil) { |node| Nodes::Dep.new(node, path) }
20
+ construct_node { |node| Nodes::Dep.new(node, path) }.reset_continue
21
21
  end
22
22
  end
23
23
  end
@@ -8,7 +8,7 @@ module Babl
8
8
  # Construct a JSON array by iterating over the current collection,
9
9
  # using the chained template for rendering each element.
10
10
  def each
11
- construct_node(key: nil, continue: nil) { |node| Nodes::Each.new(node) }
11
+ construct_node { |node| Nodes::Each.new(node) }.reset_key.reset_continue
12
12
  end
13
13
  end
14
14
  end
@@ -5,21 +5,54 @@ require 'babl/nodes'
5
5
  module Babl
6
6
  module Operators
7
7
  module Enter
8
+ KEY_QUESTIONIFIER = proc { |context|
9
+ key = context[:key]
10
+
11
+ new_key =
12
+ case key
13
+ when ::String then "#{key}?"
14
+ when ::Symbol then :"#{key}?"
15
+ else raise Errors::InvalidTemplate, "Key is expected to key a string or a symbol: #{key}"
16
+ end
17
+
18
+ context.merge(key: new_key)
19
+ }
20
+
8
21
  module DSL
9
- # Navigate to a named property of current element. The name
10
- # is inferred based on the object()
22
+ # Navigate to a property whose name is inferred based on parent object()'s key
11
23
  def enter
12
- construct_node(key: nil, continue: nil) { |node, context|
13
- key = context[:key]
14
- raise Errors::InvalidTemplate, 'No key to enter into' unless key
15
- Nodes::Nav.new(key, node)
16
- }
24
+ construct_node { |node, context|
25
+ raise Errors::InvalidTemplate, 'No key to enter into' unless context.key?(:key)
26
+
27
+ Nodes::Nav.new(context[:key], node)
28
+ }.reset_key.reset_continue
29
+ end
30
+
31
+ # Navigate to a property whose name is inferred based on parent object()'s key + '?'
32
+ def enter?
33
+ construct_context(&KEY_QUESTIONIFIER).enter
17
34
  end
18
35
 
19
36
  # Simple convenience alias
20
37
  def _
21
38
  enter
22
39
  end
40
+
41
+ # Simple convenience alias
42
+ def _?
43
+ enter?
44
+ end
45
+
46
+ protected
47
+
48
+ # Clear contextual information about current property name for the rest of the chain
49
+ def reset_key
50
+ construct_context { |context|
51
+ next context unless context.key?(:key)
52
+
53
+ context.reject { |k, _v| :key == k }
54
+ }
55
+ end
23
56
  end
24
57
  end
25
58
  end
@@ -9,15 +9,13 @@ module Babl
9
9
  # Merge multiple JSON objects (non-deep)
10
10
  def merge(*templates)
11
11
  return call(Utils::Hash::EMPTY) if templates.empty?
12
- templates = templates.map { |t| unscoped.call(t) }
12
+
13
+ templates = templates.map { |t| unscoped.reset_continue.call(t) }
13
14
 
14
15
  construct_terminal { |context|
15
16
  Nodes::Merge.new(
16
17
  templates.map { |t|
17
- t.builder.precompile(
18
- Nodes::TerminalValue.instance,
19
- context.merge(continue: nil)
20
- )
18
+ t.builder.precompile(Nodes::TerminalValue.instance, context)
21
19
  }
22
20
  )
23
21
  }
@@ -12,10 +12,11 @@ module Babl
12
12
  # is disabled for the rest of the chain.
13
13
  def nav(*path, &block)
14
14
  if path.empty?
15
- return (block ? with(unscoped, &block) : construct_node(key: nil, continue: nil) { |node| node })
15
+ return (block ? with(unscoped, &block) : reset_key.reset_continue)
16
16
  end
17
+
17
18
  property = path.first.dup.freeze
18
- construct_node(key: nil, continue: nil) { |node| Nodes::Nav.new(property, node) }
19
+ construct_node { |node| Nodes::Nav.new(property, node) }
19
20
  .nav(*path[1..-1], &block)
20
21
  end
21
22
  end
@@ -18,7 +18,7 @@ module Babl
18
18
  templates = args
19
19
  .map { |name| [name.to_sym, unscoped.nav(name)] }.to_h
20
20
  .merge(kwargs)
21
- .map { |k, v| [k, unscoped.call(v)] }
21
+ .map { |k, v| [k, unscoped.reset_continue.call(v)] }
22
22
 
23
23
  construct_terminal { |ctx|
24
24
  Nodes::Object.new(templates.map { |key, template|
@@ -26,7 +26,7 @@ module Babl
26
26
  key.to_sym,
27
27
  template.builder.precompile(
28
28
  Nodes::TerminalValue.instance,
29
- ctx.merge(key: key, continue: nil)
29
+ ctx.merge(key: key)
30
30
  )
31
31
  ]
32
32
  }.to_h)
@@ -7,7 +7,7 @@ module Babl
7
7
  module DSL
8
8
  # Navigate to the parent of the current object.
9
9
  def parent
10
- construct_node(key: nil, continue: nil) { |node| Nodes::Parent.new(node) }
10
+ construct_node { |node| Nodes::Parent.new(node) }.reset_key.reset_continue
11
11
  end
12
12
 
13
13
  protected
@@ -16,6 +16,7 @@ module Babl
16
16
  if tree.dependencies.key? Nodes::Parent::PARENT_MARKER
17
17
  raise Errors::InvalidTemplate, 'Out of context parent dependency'
18
18
  end
19
+
19
20
  super
20
21
  end
21
22
  end
@@ -12,8 +12,10 @@ module Babl
12
12
  construct_terminal { |ctx|
13
13
  lookup_context = ctx[:lookup_context]
14
14
  raise Errors::InvalidTemplate, 'Cannot use partial without lookup context' unless lookup_context
15
+
15
16
  template, new_lookup_context = lookup_context.find(current_template, partial_name)
16
17
  raise Errors::InvalidTemplate, "Cannot find partial '#{partial_name}'" unless template
18
+
17
19
  template.precompile(Nodes::TerminalValue.instance, lookup_context: new_lookup_context)
18
20
  }
19
21
  end
@@ -15,12 +15,12 @@ module Babl
15
15
 
16
16
  def named_pin(ref)
17
17
  check_pin_ref(ref)
18
- construct_node(continue: nil) { |node| Nodes::CreatePin.new(node, ref) }
18
+ construct_node { |node| Nodes::CreatePin.new(node, ref) }.reset_continue
19
19
  end
20
20
 
21
21
  def goto_pin(ref)
22
22
  check_pin_ref(ref)
23
- construct_node(key: nil, continue: nil) { |node| Nodes::GotoPin.new(node, ref) }
23
+ construct_node { |node| Nodes::GotoPin.new(node, ref) }.reset_key.reset_continue
24
24
  end
25
25
 
26
26
  protected
@@ -28,6 +28,7 @@ module Babl
28
28
  def validate(tree)
29
29
  name = tree.pinned_dependencies.keys.first
30
30
  raise Errors::InvalidTemplate, "Unresolved pin: #{name}" if name
31
+
31
32
  super
32
33
  end
33
34
 
@@ -7,6 +7,7 @@ module Babl
7
7
  # Parse BABL source into a Template
8
8
  def source(*args, &block)
9
9
  raise Errors::InvalidTemplate, 'source() expects a block xor a string' unless args.empty? ^ block.nil?
10
+
10
11
  block ||= proc { instance_eval(*args) }
11
12
  call Utils::DslProxy.eval(unscoped, &block)
12
13
  end
@@ -8,13 +8,13 @@ module Babl
8
8
  module DSL
9
9
  # Conditional switching between different templates
10
10
  def switch(conds = Utils::Hash::EMPTY)
11
- conds = conds.map { |cond, value| [unscoped.call(cond), unscoped.call(value)] }
11
+ conds = conds.map { |cond, value| [unscoped.reset_continue.call(cond), unscoped.call(value)] }
12
12
 
13
- construct_node(continue: nil) { |node, context|
13
+ construct_node { |node, context|
14
14
  nodes = conds.map { |cond, value|
15
15
  cond_node = cond.builder.precompile(
16
16
  Nodes::InternalValue.instance,
17
- context.merge(continue: nil)
17
+ context
18
18
  )
19
19
 
20
20
  value_node = value.builder.precompile(
@@ -26,7 +26,7 @@ module Babl
26
26
  }
27
27
 
28
28
  Nodes::Switch.new(nodes)
29
- }
29
+ }.reset_continue
30
30
  end
31
31
  end
32
32
  end
@@ -7,16 +7,16 @@ module Babl
7
7
  module DSL
8
8
  # Produce a value by calling the block, passing it the output value of the templates passed as argument.
9
9
  def with(*templates, &block)
10
- templates = templates.map { |t| unscoped.call(t) }
10
+ templates = templates.map { |t| unscoped.reset_continue.call(t) }
11
11
 
12
- construct_node(key: nil, continue: nil) do |node, context|
12
+ construct_node { |node, context|
13
13
  Nodes::With.new(node, templates.map { |t|
14
14
  t.builder.precompile(
15
15
  Nodes::InternalValue.instance,
16
- context.merge(continue: nil)
16
+ context
17
17
  )
18
18
  }, block)
19
- end
19
+ }.reset_key.reset_continue
20
20
  end
21
21
  end
22
22
  end
@@ -67,6 +67,7 @@ module Babl
67
67
 
68
68
  def simplify_integer_is_number
69
69
  return unless choice_set.include?(Typed::INTEGER) && choice_set.include?(Typed::NUMBER)
70
+
70
71
  AnyOf.canonicalized(choice_set - [Typed::INTEGER])
71
72
  end
72
73
 
@@ -103,6 +104,7 @@ module Babl
103
104
  # AnyOf[true, false] is just boolean
104
105
  def simplify_boolean
105
106
  return unless choice_set.include?(Primitive::TRUE) && choice_set.include?(Primitive::FALSE)
107
+
106
108
  AnyOf.canonicalized(choice_set - [Primitive::TRUE, Primitive::FALSE] + [Typed::BOOLEAN])
107
109
  end
108
110
 
@@ -113,10 +115,12 @@ module Babl
113
115
  def simplify_typed_and_static
114
116
  choice_set.each do |typed|
115
117
  next unless Typed === typed
118
+
116
119
  instances = choice_set.select { |instance|
117
120
  Primitive === instance && typed.classes.any? { |clazz| clazz === instance.value }
118
121
  }
119
122
  next if instances.empty?
123
+
120
124
  return AnyOf.canonicalized(choice_set - instances)
121
125
  end
122
126
  nil
@@ -126,8 +130,10 @@ module Babl
126
130
  # We can get rid of the former and only keep the DynArray
127
131
  def simplify_empty_array
128
132
  return unless choice_set.include?(FixedArray::EMPTY)
133
+
129
134
  choice_set.each do |other|
130
135
  next unless DynArray === other
136
+
131
137
  new_other = DynArray.new(other.item)
132
138
  return AnyOf.canonicalized(choice_set - [other, FixedArray::EMPTY] + [new_other])
133
139
  end
@@ -142,6 +148,7 @@ module Babl
142
148
 
143
149
  choice_set.each do |dyn|
144
150
  next unless DynArray === dyn
151
+
145
152
  fixed_arrays.each do |fixed|
146
153
  new_dyn = DynArray.new(dyn.item)
147
154
  return AnyOf.canonicalized(choice_set - [fixed, dyn] + [new_dyn]) if dyn.item == fixed.items.first
@@ -170,8 +177,8 @@ module Babl
170
177
  next if obj1props.any? { |name, p1|
171
178
  p2 = obj2props[name]
172
179
  next name if p2 && Primitive === p2.value &&
173
- Primitive === p1.value &&
174
- p1.value.value != p2.value.value
180
+ Primitive === p1.value &&
181
+ p1.value.value != p2.value.value
175
182
  }
176
183
 
177
184
  new_properties = (obj1props.keys + obj2props.keys).uniq.map { |name|
@@ -197,9 +204,11 @@ module Babl
197
204
  def simplify_push_down_dyn_array
198
205
  choice_set.each_with_index { |arr1, index1|
199
206
  next unless DynArray === arr1
207
+
200
208
  choice_set.each_with_index { |arr2, index2|
201
209
  break if index2 >= index1
202
210
  next unless DynArray === arr2
211
+
203
212
  new_arr = DynArray.new(AnyOf.canonicalized([arr1.item, arr2.item]))
204
213
  return AnyOf.canonicalized(choice_set - [arr1, arr2] + [new_arr])
205
214
  }
File without changes
@@ -21,10 +21,12 @@ module Babl
21
21
  def json
22
22
  { type: 'object' }.tap { |out|
23
23
  next if property_set.empty?
24
+
24
25
  out[:properties] = property_set.map { |property| [property.name, property.value.json] }.to_h
25
26
  out[:additionalProperties] = additional
26
27
  required_properties = property_set.select(&:required)
27
28
  next if required_properties.empty?
29
+
28
30
  out[:required] = property_set.select(&:required).map(&:name).map(&:to_s)
29
31
  }
30
32
  end
@@ -11,7 +11,7 @@ module Babl
11
11
  NON_PROXIED_METHODS = Set[
12
12
  :__send__, :send, :object_id, :__id__, :equal?, :instance_eval, :instance_exec,
13
13
  :respond_to?, :method, :freeze
14
- ]
14
+ ].freeze
15
15
 
16
16
  instance_methods.each do |method|
17
17
  undef_method(method) unless NON_PROXIED_METHODS.include?(method)
@@ -5,10 +5,11 @@ module Babl
5
5
  EMPTY = {}.freeze
6
6
 
7
7
  class << self
8
- def deep_merge(*hashes)
8
+ def deep_merge(hashes)
9
9
  filtered_hashes = hashes.reject(&:empty?)
10
10
  return EMPTY if filtered_hashes.empty?
11
11
  return filtered_hashes.first if filtered_hashes.size == 1
12
+
12
13
  filtered_hashes.reduce({}) { |out, hash| deep_merge_inplace(out, hash) }
13
14
  end
14
15
 
@@ -45,6 +45,7 @@ module Babl
45
45
  class << self
46
46
  def with(hash = Utils::Hash::EMPTY)
47
47
  raise ::ArgumentError unless ::Hash === hash && (hash.keys - self::FIELDS).empty?
48
+
48
49
  new(*self::FIELDS.map { |f| hash.fetch(f) })
49
50
  end
50
51
 
@@ -1,4 +1,4 @@
1
1
  # frozen_string_literal: true
2
2
  module Babl
3
- VERSION = '0.5.9'
3
+ VERSION = '0.6.0'
4
4
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: babl-json
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.5.9
4
+ version: 0.6.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Frederic Terrazzoni
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2018-09-03 00:00:00.000000000 Z
11
+ date: 2018-10-08 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: coveralls
@@ -187,6 +187,7 @@ files:
187
187
  - lib/babl/schema/anything.rb
188
188
  - lib/babl/schema/dyn_array.rb
189
189
  - lib/babl/schema/fixed_array.rb
190
+ - lib/babl/schema/json_schema.rb
190
191
  - lib/babl/schema/object.rb
191
192
  - lib/babl/schema/primitive.rb
192
193
  - lib/babl/schema/typed.rb