axiom-arango-adapter 0.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (125) hide show
  1. data/.gitignore +4 -0
  2. data/.rspec +5 -0
  3. data/.travis.yml +25 -0
  4. data/Gemfile +9 -0
  5. data/Gemfile.devtools +57 -0
  6. data/Guardfile +18 -0
  7. data/LICENSE +20 -0
  8. data/README.md +92 -0
  9. data/Rakefile +2 -0
  10. data/TODO +1 -0
  11. data/axiom-arango-adapter.gemspec +26 -0
  12. data/config/flay.yml +3 -0
  13. data/config/flog.yml +2 -0
  14. data/config/mutant.yml +3 -0
  15. data/config/roodi.yml +18 -0
  16. data/config/site.reek +102 -0
  17. data/config/yardstick.yml +2 -0
  18. data/lib/axiom-arango-adapter.rb +2 -0
  19. data/lib/axiom/adapter/arango.rb +40 -0
  20. data/lib/axiom/adapter/arango/adapter.rb +68 -0
  21. data/lib/axiom/adapter/arango/gateway.rb +356 -0
  22. data/lib/axiom/adapter/arango/reader.rb +98 -0
  23. data/lib/axiom/adapter/arango/visitor.rb +100 -0
  24. data/lib/axiom/adapter/arango/visitor/aggregate.rb +132 -0
  25. data/lib/axiom/adapter/arango/visitor/attribute.rb +26 -0
  26. data/lib/axiom/adapter/arango/visitor/binary.rb +77 -0
  27. data/lib/axiom/adapter/arango/visitor/for.rb +75 -0
  28. data/lib/axiom/adapter/arango/visitor/for/base.rb +40 -0
  29. data/lib/axiom/adapter/arango/visitor/for/binary.rb +108 -0
  30. data/lib/axiom/adapter/arango/visitor/for/binary/join.rb +131 -0
  31. data/lib/axiom/adapter/arango/visitor/for/binary/product.rb +48 -0
  32. data/lib/axiom/adapter/arango/visitor/for/summarization.rb +113 -0
  33. data/lib/axiom/adapter/arango/visitor/for/unary.rb +29 -0
  34. data/lib/axiom/adapter/arango/visitor/for/unary/extension.rb +72 -0
  35. data/lib/axiom/adapter/arango/visitor/for/unary/limit.rb +31 -0
  36. data/lib/axiom/adapter/arango/visitor/for/unary/offset.rb +34 -0
  37. data/lib/axiom/adapter/arango/visitor/for/unary/order.rb +63 -0
  38. data/lib/axiom/adapter/arango/visitor/for/unary/projection.rb +32 -0
  39. data/lib/axiom/adapter/arango/visitor/for/unary/rename.rb +88 -0
  40. data/lib/axiom/adapter/arango/visitor/for/unary/restriction.rb +42 -0
  41. data/lib/axiom/adapter/arango/visitor/function.rb +33 -0
  42. data/lib/axiom/adapter/arango/visitor/header.rb +55 -0
  43. data/lib/axiom/adapter/arango/visitor/literal.rb +26 -0
  44. data/lib/axiom/adapter/arango/visitor/reverse.rb +27 -0
  45. data/spec/fuzzer.rb +53 -0
  46. data/spec/integration/arango/read_spec.rb +166 -0
  47. data/spec/setup/arangodb.sh +61 -0
  48. data/spec/setup/run.sh +23 -0
  49. data/spec/shared/aql_behavior.rb +7 -0
  50. data/spec/shared/binary_relation_method_behaviour.rb +53 -0
  51. data/spec/shared/unary_relation_method_behaviour.rb +21 -0
  52. data/spec/spec_helper.rb +51 -0
  53. data/spec/support/ice_nine_config.rb +6 -0
  54. data/spec/unit/axiom/adapter/arango/adapter/class_methods/new_spec.rb +24 -0
  55. data/spec/unit/axiom/adapter/arango/adapter/gateway_spec.rb +11 -0
  56. data/spec/unit/axiom/adapter/arango/adapter/reader_spec.rb +12 -0
  57. data/spec/unit/axiom/adapter/arango/gateway/class_methods/new_spec.rb +15 -0
  58. data/spec/unit/axiom/adapter/arango/gateway/difference_spec.rb +16 -0
  59. data/spec/unit/axiom/adapter/arango/gateway/drop_spec.rb +20 -0
  60. data/spec/unit/axiom/adapter/arango/gateway/each_spec.rb +78 -0
  61. data/spec/unit/axiom/adapter/arango/gateway/extend_spec.rb +26 -0
  62. data/spec/unit/axiom/adapter/arango/gateway/intersect_spec.rb +16 -0
  63. data/spec/unit/axiom/adapter/arango/gateway/join_spec.rb +43 -0
  64. data/spec/unit/axiom/adapter/arango/gateway/materialize_spec.rb +26 -0
  65. data/spec/unit/axiom/adapter/arango/gateway/optimize_spec.rb +22 -0
  66. data/spec/unit/axiom/adapter/arango/gateway/product_spec.rb +16 -0
  67. data/spec/unit/axiom/adapter/arango/gateway/project_spec.rb +20 -0
  68. data/spec/unit/axiom/adapter/arango/gateway/remove_spec.rb +20 -0
  69. data/spec/unit/axiom/adapter/arango/gateway/rename_spec.rb +20 -0
  70. data/spec/unit/axiom/adapter/arango/gateway/respond_to_predicate_spec.rb +28 -0
  71. data/spec/unit/axiom/adapter/arango/gateway/restrict_spec.rb +26 -0
  72. data/spec/unit/axiom/adapter/arango/gateway/reverse_spec.rb +20 -0
  73. data/spec/unit/axiom/adapter/arango/gateway/sort_by_spec.rb +26 -0
  74. data/spec/unit/axiom/adapter/arango/gateway/summarize_spec.rb +150 -0
  75. data/spec/unit/axiom/adapter/arango/gateway/take_spec.rb +20 -0
  76. data/spec/unit/axiom/adapter/arango/gateway/unhandled_message_spec.rb +17 -0
  77. data/spec/unit/axiom/adapter/arango/gateway/union_spec.rb +16 -0
  78. data/spec/unit/axiom/adapter/arango/reader/each_spec.rb +43 -0
  79. data/spec/unit/axiom/adapter/arango/visitor/aggregate/count/root_spec.rb +15 -0
  80. data/spec/unit/axiom/adapter/arango/visitor/aggregate/count_spec.rb +15 -0
  81. data/spec/unit/axiom/adapter/arango/visitor/aggregate/maximum/root_spec.rb +15 -0
  82. data/spec/unit/axiom/adapter/arango/visitor/aggregate/maximum_spec.rb +15 -0
  83. data/spec/unit/axiom/adapter/arango/visitor/aggregate/minimum/root_spec.rb +15 -0
  84. data/spec/unit/axiom/adapter/arango/visitor/aggregate/minimum_spec.rb +15 -0
  85. data/spec/unit/axiom/adapter/arango/visitor/aggregate/sum/root_spec.rb +15 -0
  86. data/spec/unit/axiom/adapter/arango/visitor/aggregate/sum_spec.rb +15 -0
  87. data/spec/unit/axiom/adapter/arango/visitor/attribute/root_spec.rb +12 -0
  88. data/spec/unit/axiom/adapter/arango/visitor/binary/root_spec.rb +15 -0
  89. data/spec/unit/axiom/adapter/arango/visitor/class_methods/build_spec.rb +37 -0
  90. data/spec/unit/axiom/adapter/arango/visitor/class_methods/run_spec.rb +26 -0
  91. data/spec/unit/axiom/adapter/arango/visitor/for/base/root_spec.rb +13 -0
  92. data/spec/unit/axiom/adapter/arango/visitor/for/binary/join/local_name_spec.rb +20 -0
  93. data/spec/unit/axiom/adapter/arango/visitor/for/binary/join/right/root_spec.rb +14 -0
  94. data/spec/unit/axiom/adapter/arango/visitor/for/binary/join/root_spec.rb +14 -0
  95. data/spec/unit/axiom/adapter/arango/visitor/for/binary/join_spec.rb +14 -0
  96. data/spec/unit/axiom/adapter/arango/visitor/for/binary/local_name_spec.rb +14 -0
  97. data/spec/unit/axiom/adapter/arango/visitor/for/binary/product/right/root_spec.rb +13 -0
  98. data/spec/unit/axiom/adapter/arango/visitor/for/binary/product/root_spec.rb +13 -0
  99. data/spec/unit/axiom/adapter/arango/visitor/for/binary/product_spec.rb +13 -0
  100. data/spec/unit/axiom/adapter/arango/visitor/for/binary/right/join_spec.rb +14 -0
  101. data/spec/unit/axiom/adapter/arango/visitor/for/binary/right/local_name_spec.rb +13 -0
  102. data/spec/unit/axiom/adapter/arango/visitor/for/binary/right/product_spec.rb +13 -0
  103. data/spec/unit/axiom/adapter/arango/visitor/for/local_name_spec.rb +21 -0
  104. data/spec/unit/axiom/adapter/arango/visitor/for/root_spec.rb +29 -0
  105. data/spec/unit/axiom/adapter/arango/visitor/for/summarization/root_spec.rb +53 -0
  106. data/spec/unit/axiom/adapter/arango/visitor/for/unary/extension/root_spec.rb +13 -0
  107. data/spec/unit/axiom/adapter/arango/visitor/for/unary/extension_spec.rb +13 -0
  108. data/spec/unit/axiom/adapter/arango/visitor/for/unary/limit/root_spec.rb +17 -0
  109. data/spec/unit/axiom/adapter/arango/visitor/for/unary/limit_spec.rb +17 -0
  110. data/spec/unit/axiom/adapter/arango/visitor/for/unary/offset/root_spec.rb +17 -0
  111. data/spec/unit/axiom/adapter/arango/visitor/for/unary/offset_spec.rb +17 -0
  112. data/spec/unit/axiom/adapter/arango/visitor/for/unary/order/root_spec.rb +14 -0
  113. data/spec/unit/axiom/adapter/arango/visitor/for/unary/order_spec.rb +14 -0
  114. data/spec/unit/axiom/adapter/arango/visitor/for/unary/projection/root_spec.rb +13 -0
  115. data/spec/unit/axiom/adapter/arango/visitor/for/unary/projection_spec.rb +13 -0
  116. data/spec/unit/axiom/adapter/arango/visitor/for/unary/rename/root_spec.rb +13 -0
  117. data/spec/unit/axiom/adapter/arango/visitor/for/unary/rename_spec.rb +13 -0
  118. data/spec/unit/axiom/adapter/arango/visitor/for/unary/restriction/root_spec.rb +14 -0
  119. data/spec/unit/axiom/adapter/arango/visitor/for/unary/restriction_spec.rb +14 -0
  120. data/spec/unit/axiom/adapter/arango/visitor/header/root_spec.rb +12 -0
  121. data/spec/unit/axiom/adapter/arango/visitor/literal/root_spec.rb +10 -0
  122. data/spec/unit/axiom/adapter/arango/visitor/static/root_spec.rb +19 -0
  123. data/spec/unit/axiom/adapter/arango/visitor/visit_spec.rb +52 -0
  124. data/spec/unit/axiom/adapter/arango/visitor/visitor_spec.rb +39 -0
  125. metadata +379 -0
@@ -0,0 +1,100 @@
1
+ module Axiom
2
+ module Adapter
3
+ module Arango
4
+ # Base class for relation specific visitors
5
+ class Visitor
6
+ include AbstractType, Adamantium::Flat, Concord.new(:input, :context), AQL
7
+
8
+ abstract_method :root
9
+
10
+ # Return an AQL node from the relation
11
+ #
12
+ # @param [Relation] relation
13
+ #
14
+ # @return [AQL::Node]
15
+ #
16
+ # @api private
17
+ #
18
+ def self.run(relation)
19
+ build(relation).root
20
+ end
21
+
22
+ REGISTRY = {}
23
+
24
+ # Register handler for the axiom relation
25
+ #
26
+ # @param [Class] klass
27
+ #
28
+ # @return [undefined]
29
+ #
30
+ # @api private
31
+ #
32
+ def self.handle(klass)
33
+ REGISTRY[klass]=self
34
+ end
35
+
36
+ private_class_method :handle
37
+
38
+ # Return visitor for relation and context
39
+ #
40
+ # @param [Relation] relation
41
+ # @param [Visitor] context
42
+ #
43
+ # @return [Visitor]
44
+ #
45
+ # @api private
46
+ #
47
+ def self.build(relation, context = nil)
48
+ REGISTRY.fetch(relation.class).new(relation, context)
49
+ end
50
+
51
+ protected
52
+
53
+ # Return depth
54
+ #
55
+ # @return [Integer]
56
+ #
57
+ # @api private
58
+ #
59
+ def depth
60
+ context = self.context
61
+ if context
62
+ context.depth + 1
63
+ else
64
+ 0
65
+ end
66
+ end
67
+ memoize :depth
68
+
69
+ private
70
+
71
+ # Return a visitor for the given relation
72
+ #
73
+ # @param [Relation] relation
74
+ # @param [Visitor] context
75
+ #
76
+ # @return [Visitor]
77
+ #
78
+ # @api private
79
+ #
80
+ def visitor(relation, context = self)
81
+ self.class.build(relation, context)
82
+ end
83
+
84
+ # Return an AQL node from visiting axiom a given relation
85
+ #
86
+ # @param [Relation] relation
87
+ # @param [Visitor] context
88
+ #
89
+ # @return [AQL::Node]
90
+ #
91
+ # @api private
92
+ #
93
+ def visit(relation, context = self)
94
+ visitor(relation, context).root
95
+ end
96
+
97
+ end
98
+ end
99
+ end
100
+ end
@@ -0,0 +1,132 @@
1
+ module Axiom
2
+ module Adapter
3
+ module Arango
4
+ class Visitor
5
+ # Base class for aggregate visitors
6
+ class Aggregate < self
7
+
8
+ LOCAL_NAME = AQL.name_node('aggregate')
9
+
10
+ # Return root AQL node
11
+ #
12
+ # @return [AQL::Node]
13
+ #
14
+ # @api private
15
+ #
16
+ def root
17
+ Node::Call.new(self.class::FUNCTION, [mapped])
18
+ end
19
+ memoize :root
20
+
21
+ private
22
+
23
+ # Return attribute name used for mapping
24
+ #
25
+ # @return [AQL::Node::Name]
26
+ #
27
+ # @api private
28
+ #
29
+ def map_attribute_name
30
+ Node::Name.new(input.operand.name.to_s)
31
+ end
32
+ memoize :map_attribute_name
33
+
34
+ # Return summarization attribute
35
+ #
36
+ # @return [AQL::Node::Attribute]
37
+ #
38
+ # @api private
39
+ #
40
+ def summarization_attribute
41
+ Node::Attribute.new(For::Summarization::LOCAL_NAME, map_attribute_name)
42
+ end
43
+
44
+ # Return attribute used for mapping
45
+ #
46
+ # @return [AQL::Node::Attribute]
47
+ #
48
+ # @api private
49
+ #
50
+ def map_attribute
51
+ Node::Attribute.new(LOCAL_NAME, summarization_attribute)
52
+ end
53
+ memoize :map_attribute
54
+
55
+ # Return mapped collection
56
+ #
57
+ # @return [AQL::Node::Operation::For]
58
+ #
59
+ # @api private
60
+ #
61
+ def mapped
62
+ Node::Operation::For.new(
63
+ LOCAL_NAME,
64
+ For::Summarization::COLLECT_NAME,
65
+ map_body
66
+ )
67
+ end
68
+
69
+ # Return map filter operation
70
+ #
71
+ # @return [AQL::Node::Operation::Unary::Filter]
72
+ #
73
+ # @api private
74
+ #
75
+ def map_filter
76
+ Node::Operation::Unary::Filter.new(
77
+ Node::Operator::Binary::Inequality.new(map_attribute, Node::Literal::Singleton::NULL)
78
+ )
79
+ end
80
+
81
+ # Return map return operation
82
+ #
83
+ # @return [AQL::Node::Operation::Unary::Return]
84
+ #
85
+ # @api private
86
+ #
87
+ def map_return
88
+ Node::Operation::Unary::Return.new(map_attribute)
89
+ end
90
+
91
+ # Return map body
92
+ #
93
+ # @return [AQL::Node::Block]
94
+ #
95
+ # @api private
96
+ #
97
+ def map_body
98
+ Node::Block.new([
99
+ map_filter,
100
+ map_return
101
+ ])
102
+ end
103
+
104
+ # Visitor for count aggregate
105
+ class Count < self
106
+ FUNCTION = 'LENGTH'.freeze
107
+ handle(Axiom::Aggregate::Count)
108
+ end
109
+
110
+ # Visitor for sum aggregate
111
+ class Sum < self
112
+ FUNCTION = 'SUM'.freeze
113
+ handle(Axiom::Aggregate::Sum)
114
+ end
115
+
116
+ # Visitor for maximum aggregate
117
+ class Maximum < self
118
+ FUNCTION = 'MAX'.freeze
119
+ handle(Axiom::Aggregate::Maximum)
120
+ end
121
+
122
+ # Visitor for minimum aggregate
123
+ class Minimum < self
124
+ FUNCTION = 'MIN'.freeze
125
+ handle(Axiom::Aggregate::Minimum)
126
+ end
127
+
128
+ end
129
+ end
130
+ end
131
+ end
132
+ end
@@ -0,0 +1,26 @@
1
+ module Axiom
2
+ module Adapter
3
+ module Arango
4
+ class Visitor
5
+ # Visitor for attribute accesses
6
+ class Attribute < self
7
+
8
+ handle(Axiom::Attribute::String)
9
+ handle(Axiom::Attribute::Integer)
10
+
11
+ # Return root AQL node
12
+ #
13
+ # @return [AQL::Node]
14
+ #
15
+ # @api private
16
+ #
17
+ def root
18
+ Node::Attribute.new(context.local_name, AQL.name_node(input.name))
19
+ end
20
+ memoize :root
21
+
22
+ end
23
+ end
24
+ end
25
+ end
26
+ end
@@ -0,0 +1,77 @@
1
+ module Axiom
2
+ module Adapter
3
+ module Arango
4
+ class Visitor
5
+ # Visitor for binary AQL nodes
6
+ class Binary < self
7
+
8
+ MAPPING = {
9
+ Function::Numeric::Multiplication => AQL::Node::Operator::Binary::Multiplication,
10
+ Function::Numeric::Addition => AQL::Node::Operator::Binary::Addition,
11
+ Function::Numeric::Division => AQL::Node::Operator::Binary::Division,
12
+ Function::Numeric::Modulo => AQL::Node::Operator::Binary::Modulo,
13
+ Function::Numeric::Subtraction => AQL::Node::Operator::Binary::Subtraction,
14
+ Function::Predicate::GreaterThanOrEqualTo => AQL::Node::Operator::Binary::GreaterThanOrEqualTo,
15
+ Function::Predicate::GreaterThan => AQL::Node::Operator::Binary::GreaterThan,
16
+ Function::Predicate::LessThan => AQL::Node::Operator::Binary::LessThan,
17
+ Function::Predicate::LessThanOrEqualTo => AQL::Node::Operator::Binary::LessThanOrEqualTo,
18
+ Function::Connective::Disjunction => AQL::Node::Operator::Binary::Or,
19
+ Function::Connective::Conjunction => AQL::Node::Operator::Binary::And,
20
+ Function::Predicate::Equality => AQL::Node::Operator::Binary::Equality,
21
+ Function::Predicate::Inequality => AQL::Node::Operator::Binary::Inequality
22
+ }.freeze
23
+
24
+ MAPPING.each_key do |klass|
25
+ handle(klass)
26
+ end
27
+
28
+ # Return local name
29
+ #
30
+ # @return [Node::Name]
31
+ #
32
+ # @api private
33
+ #
34
+ def local_name
35
+ context.local_name
36
+ end
37
+ memoize :local_name
38
+
39
+ # Return root AQL node
40
+ #
41
+ # @return [AQL::Node]
42
+ #
43
+ # @api private
44
+ #
45
+ def root
46
+ klass = MAPPING.fetch(input.class)
47
+ klass.new(left, right)
48
+ end
49
+ memoize :root
50
+
51
+ private
52
+
53
+ # Return left AQL node
54
+ #
55
+ # @return [AQL::Node]
56
+ #
57
+ # @api private
58
+ #
59
+ def left
60
+ visit(input.left)
61
+ end
62
+
63
+ # Return right AQL node
64
+ #
65
+ # @return [AQL::Node]
66
+ #
67
+ # @api private
68
+ #
69
+ def right
70
+ visit(input.right)
71
+ end
72
+
73
+ end
74
+ end
75
+ end
76
+ end
77
+ end
@@ -0,0 +1,75 @@
1
+ module Axiom
2
+ module Adapter
3
+ module Arango
4
+ class Visitor
5
+ # Base class for visitors that emit for statements
6
+ class For < self
7
+
8
+ # Return root AQL AST
9
+ #
10
+ # @return [AQL::Node]
11
+ #
12
+ # @api private
13
+ #
14
+ def root
15
+ Node::Operation::For.new(local_name, source, body)
16
+ end
17
+ memoize :root
18
+
19
+ # Return local name
20
+ #
21
+ # @return [AQL::Node::Name]
22
+ #
23
+ # @api private
24
+ #
25
+ def local_name
26
+ self.class::LOCAL_NAME
27
+ end
28
+
29
+ private
30
+
31
+ # Return source
32
+ #
33
+ # @return [AQL::Node]
34
+ #
35
+ # @api private
36
+ #
37
+ def source
38
+ visit(input.operand)
39
+ end
40
+
41
+ # Return body
42
+ #
43
+ # @return [AQL::Node]
44
+ #
45
+ # @api private
46
+ #
47
+ def body
48
+ return_operation
49
+ end
50
+
51
+ # Return return operation
52
+ #
53
+ # @return [AQL::Node::Operation::Return]
54
+ #
55
+ # @api private
56
+ #
57
+ def return_operation
58
+ Node::Operation::Unary::Return.new(return_value)
59
+ end
60
+
61
+ # Return return value
62
+ #
63
+ # @return [AQL::Node::Literal::Composed::Document]
64
+ #
65
+ # @api private
66
+ #
67
+ def return_value
68
+ visit(input.header)
69
+ end
70
+
71
+ end
72
+ end
73
+ end
74
+ end
75
+ end
@@ -0,0 +1,40 @@
1
+ module Axiom
2
+ module Adapter
3
+ module Arango
4
+ class Visitor
5
+ class For
6
+ # Visitor handling axiom base relations
7
+ class Base < self
8
+
9
+ handle(Axiom::Relation::Base)
10
+
11
+ LOCAL_NAME = AQL.name_node('base')
12
+
13
+ private
14
+
15
+ # Return source of iteration
16
+ #
17
+ # @return [AQL::Node::Name]
18
+ #
19
+ # @api private
20
+ #
21
+ def source
22
+ AQL.name_node(input.name)
23
+ end
24
+
25
+ # Return body of for statement
26
+ #
27
+ # @return [AQL::Node::Body]
28
+ #
29
+ # @api private
30
+ #
31
+ def body
32
+ Node::Block.new([return_operation])
33
+ end
34
+
35
+ end
36
+ end
37
+ end
38
+ end
39
+ end
40
+ end