mirror of
https://github.com/rails/rails.git
synced 2022-11-09 12:12:34 -05:00
Merge pull request #103 from alexstaubo/master
ANSI SQL2003 window functions
This commit is contained in:
commit
b757de114c
17 changed files with 458 additions and 3 deletions
|
@ -3,6 +3,7 @@ require 'arel/factory_methods'
|
|||
|
||||
require 'arel/expressions'
|
||||
require 'arel/predications'
|
||||
require 'arel/window_predications'
|
||||
require 'arel/math'
|
||||
require 'arel/alias_predication'
|
||||
require 'arel/order_predications'
|
||||
|
|
|
@ -19,5 +19,9 @@ module Arel
|
|||
def average
|
||||
Nodes::Avg.new [self], Nodes::SqlLiteral.new('avg_id')
|
||||
end
|
||||
|
||||
def extract field
|
||||
Nodes::Extract.new [self], field
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
|
@ -26,6 +26,7 @@ require 'arel/nodes/join_source'
|
|||
require 'arel/nodes/delete_statement'
|
||||
require 'arel/nodes/table_alias'
|
||||
require 'arel/nodes/infix_operation'
|
||||
require 'arel/nodes/over'
|
||||
|
||||
# nary
|
||||
require 'arel/nodes/and'
|
||||
|
@ -35,9 +36,13 @@ require 'arel/nodes/and'
|
|||
# We should make Function a Unary node and deprecate the use of "aliaz"
|
||||
require 'arel/nodes/function'
|
||||
require 'arel/nodes/count'
|
||||
require 'arel/nodes/extract'
|
||||
require 'arel/nodes/values'
|
||||
require 'arel/nodes/named_function'
|
||||
|
||||
# windows
|
||||
require 'arel/nodes/window'
|
||||
|
||||
# joins
|
||||
require 'arel/nodes/inner_join'
|
||||
require 'arel/nodes/outer_join'
|
||||
|
|
23
lib/arel/nodes/extract.rb
Normal file
23
lib/arel/nodes/extract.rb
Normal file
|
@ -0,0 +1,23 @@
|
|||
module Arel
|
||||
module Nodes
|
||||
|
||||
class Extract < Arel::Nodes::Unary
|
||||
include Arel::Expression
|
||||
include Arel::Predications
|
||||
|
||||
attr_accessor :field
|
||||
attr_accessor :alias
|
||||
|
||||
def initialize expr, field, aliaz = nil
|
||||
super(expr)
|
||||
@field = field
|
||||
@alias = aliaz && SqlLiteral.new(aliaz)
|
||||
end
|
||||
|
||||
def as aliaz
|
||||
self.alias = SqlLiteral.new(aliaz)
|
||||
self
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
|
@ -3,6 +3,7 @@ module Arel
|
|||
class Function < Arel::Nodes::Node
|
||||
include Arel::Expression
|
||||
include Arel::Predications
|
||||
include Arel::WindowPredications
|
||||
attr_accessor :expressions, :alias, :distinct
|
||||
|
||||
def initialize expr, aliaz = nil
|
||||
|
|
15
lib/arel/nodes/over.rb
Normal file
15
lib/arel/nodes/over.rb
Normal file
|
@ -0,0 +1,15 @@
|
|||
module Arel
|
||||
module Nodes
|
||||
|
||||
class Over < Binary
|
||||
include Arel::AliasPredication
|
||||
|
||||
def initialize(left, right = nil)
|
||||
super(left, right)
|
||||
end
|
||||
|
||||
def operator; 'OVER' end
|
||||
end
|
||||
|
||||
end
|
||||
end
|
|
@ -1,7 +1,7 @@
|
|||
module Arel
|
||||
module Nodes
|
||||
class SelectCore < Arel::Nodes::Node
|
||||
attr_accessor :top, :projections, :wheres, :groups
|
||||
attr_accessor :top, :projections, :wheres, :groups, :windows
|
||||
attr_accessor :having, :source, :set_quantifier
|
||||
|
||||
def initialize
|
||||
|
@ -14,6 +14,7 @@ module Arel
|
|||
@wheres = []
|
||||
@groups = []
|
||||
@having = nil
|
||||
@windows = []
|
||||
end
|
||||
|
||||
def from
|
||||
|
@ -34,6 +35,7 @@ module Arel
|
|||
@wheres = @wheres.clone
|
||||
@groups = @groups.clone
|
||||
@having = @having.clone if @having
|
||||
@windows = @windows.clone
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
78
lib/arel/nodes/window.rb
Normal file
78
lib/arel/nodes/window.rb
Normal file
|
@ -0,0 +1,78 @@
|
|||
module Arel
|
||||
module Nodes
|
||||
class Window < Arel::Nodes::Node
|
||||
include Arel::Expression
|
||||
attr_accessor :orders, :framing
|
||||
|
||||
def initialize
|
||||
@orders = []
|
||||
end
|
||||
|
||||
def order *expr
|
||||
# FIXME: We SHOULD NOT be converting these to SqlLiteral automatically
|
||||
@orders.concat expr.map { |x|
|
||||
String === x || Symbol === x ? Nodes::SqlLiteral.new(x.to_s) : x
|
||||
}
|
||||
self
|
||||
end
|
||||
|
||||
def frame(expr)
|
||||
raise ArgumentError, "Window frame cannot be set more than once" if @frame
|
||||
@framing = expr
|
||||
end
|
||||
|
||||
def rows(expr = nil)
|
||||
frame(Rows.new(expr))
|
||||
end
|
||||
|
||||
def range(expr = nil)
|
||||
frame(Range.new(expr))
|
||||
end
|
||||
|
||||
def initialize_copy other
|
||||
super
|
||||
@orders = @orders.map { |x| x.clone }
|
||||
end
|
||||
end
|
||||
|
||||
class NamedWindow < Window
|
||||
attr_accessor :name
|
||||
|
||||
def initialize name
|
||||
super()
|
||||
@name = name
|
||||
end
|
||||
|
||||
def initialize_copy other
|
||||
super
|
||||
@name = other.name.clone
|
||||
end
|
||||
end
|
||||
|
||||
class Rows < Unary
|
||||
def initialize(expr = nil)
|
||||
super(expr)
|
||||
end
|
||||
end
|
||||
|
||||
class Range < Unary
|
||||
def initialize(expr = nil)
|
||||
super(expr)
|
||||
end
|
||||
end
|
||||
|
||||
class CurrentRow < Arel::Nodes::Node; end
|
||||
|
||||
class Preceding < Unary
|
||||
def initialize(expr = nil)
|
||||
super(expr)
|
||||
end
|
||||
end
|
||||
|
||||
class Following < Unary
|
||||
def initialize(expr = nil)
|
||||
super(expr)
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
|
@ -126,6 +126,12 @@ module Arel
|
|||
self
|
||||
end
|
||||
|
||||
def window name
|
||||
window = Nodes::NamedWindow.new(name)
|
||||
@ctx.windows.push window
|
||||
window
|
||||
end
|
||||
|
||||
def project *projections
|
||||
# FIXME: converting these to SQLLiterals is probably not good, but
|
||||
# rails tests require it.
|
||||
|
|
|
@ -110,6 +110,7 @@ module Arel
|
|||
alias :visit_Arel_Nodes_Node :terminal
|
||||
alias :visit_Arel_Nodes_SqlLiteral :terminal
|
||||
alias :visit_Arel_Nodes_BindParam :terminal
|
||||
alias :visit_Arel_Nodes_Window :terminal
|
||||
alias :visit_Arel_SqlLiteral :terminal
|
||||
alias :visit_BigDecimal :terminal
|
||||
alias :visit_Bignum :terminal
|
||||
|
@ -136,6 +137,7 @@ module Arel
|
|||
visit o.source
|
||||
visit o.wheres
|
||||
visit o.groups
|
||||
visit o.windows
|
||||
visit o.having
|
||||
end
|
||||
|
||||
|
|
|
@ -74,6 +74,23 @@ module Arel
|
|||
alias :visit_Arel_Nodes_On :unary
|
||||
alias :visit_Arel_Nodes_Top :unary
|
||||
alias :visit_Arel_Nodes_UnqualifiedColumn :unary
|
||||
alias :visit_Arel_Nodes_Preceding :unary
|
||||
alias :visit_Arel_Nodes_Following :unary
|
||||
alias :visit_Arel_Nodes_Rows :unary
|
||||
alias :visit_Arel_Nodes_Range :unary
|
||||
|
||||
def window o
|
||||
visit_edge o, "orders"
|
||||
visit_edge o, "framing"
|
||||
end
|
||||
alias :visit_Arel_Nodes_Window :window
|
||||
|
||||
def named_window o
|
||||
visit_edge o, "orders"
|
||||
visit_edge o, "framing"
|
||||
visit_edge o, "name"
|
||||
end
|
||||
alias :visit_Arel_Nodes_NamedWindow :named_window
|
||||
|
||||
def function o
|
||||
visit_edge o, "expressions"
|
||||
|
@ -86,6 +103,12 @@ module Arel
|
|||
alias :visit_Arel_Nodes_Avg :function
|
||||
alias :visit_Arel_Nodes_Sum :function
|
||||
|
||||
def extract o
|
||||
visit_edge o, "expressions"
|
||||
visit_edge o, "alias"
|
||||
end
|
||||
alias :visit_Arel_Nodes_Extract :extract
|
||||
|
||||
def visit_Arel_Nodes_NamedFunction o
|
||||
visit_edge o, "name"
|
||||
visit_edge o, "expressions"
|
||||
|
@ -103,6 +126,7 @@ module Arel
|
|||
visit_edge o, "source"
|
||||
visit_edge o, "projections"
|
||||
visit_edge o, "wheres"
|
||||
visit_edge o, "windows"
|
||||
end
|
||||
|
||||
def visit_Arel_Nodes_SelectStatement o
|
||||
|
@ -159,6 +183,7 @@ module Arel
|
|||
alias :visit_Arel_Nodes_NotEqual :binary
|
||||
alias :visit_Arel_Nodes_NotIn :binary
|
||||
alias :visit_Arel_Nodes_Or :binary
|
||||
alias :visit_Arel_Nodes_Over :binary
|
||||
|
||||
def visit_String o
|
||||
@node_stack.last.fields << o
|
||||
|
|
|
@ -136,6 +136,7 @@ key on UpdateManager using UpdateManager#key=
|
|||
("WHERE #{o.wheres.map { |x| visit x }.join ' AND ' }" unless o.wheres.empty?),
|
||||
("GROUP BY #{o.groups.map { |x| visit x }.join ', ' }" unless o.groups.empty?),
|
||||
(visit(o.having) if o.having),
|
||||
("WINDOW #{o.windows.map { |x| visit x }.join ', ' }" unless o.windows.empty?)
|
||||
].compact.join ' '
|
||||
end
|
||||
|
||||
|
@ -175,6 +176,59 @@ key on UpdateManager using UpdateManager#key=
|
|||
"( #{visit o.left} EXCEPT #{visit o.right} )"
|
||||
end
|
||||
|
||||
def visit_Arel_Nodes_NamedWindow o
|
||||
"#{quote_column_name o.name} AS #{visit_Arel_Nodes_Window o}"
|
||||
end
|
||||
|
||||
def visit_Arel_Nodes_Window o
|
||||
s = [
|
||||
("ORDER BY #{o.orders.map { |x| visit(x) }.join(', ')}" unless o.orders.empty?),
|
||||
(visit o.framing if o.framing)
|
||||
].compact.join ' '
|
||||
"(#{s})"
|
||||
end
|
||||
|
||||
def visit_Arel_Nodes_Rows o
|
||||
if o.expr
|
||||
"ROWS #{visit o.expr}"
|
||||
else
|
||||
"ROWS"
|
||||
end
|
||||
end
|
||||
|
||||
def visit_Arel_Nodes_Range o
|
||||
if o.expr
|
||||
"RANGE #{visit o.expr}"
|
||||
else
|
||||
"RANGE"
|
||||
end
|
||||
end
|
||||
|
||||
def visit_Arel_Nodes_Preceding o
|
||||
"#{o.expr ? visit(o.expr) : 'UNBOUNDED'} PRECEDING"
|
||||
end
|
||||
|
||||
def visit_Arel_Nodes_Following o
|
||||
"#{o.expr ? visit(o.expr) : 'UNBOUNDED'} FOLLOWING"
|
||||
end
|
||||
|
||||
def visit_Arel_Nodes_CurrentRow o
|
||||
"CURRENT ROW"
|
||||
end
|
||||
|
||||
def visit_Arel_Nodes_Over o
|
||||
case o.right
|
||||
when nil
|
||||
"#{visit o.left} OVER ()"
|
||||
when Arel::Nodes::SqlLiteral
|
||||
"#{visit o.left} OVER #{visit o.right}"
|
||||
when String, Symbol
|
||||
"#{visit o.left} OVER #{quote_column_name o.right.to_s}"
|
||||
else
|
||||
"#{visit o.left} OVER #{visit o.right}"
|
||||
end
|
||||
end
|
||||
|
||||
def visit_Arel_Nodes_Having o
|
||||
"HAVING #{visit o.expr}"
|
||||
end
|
||||
|
@ -218,6 +272,10 @@ key on UpdateManager using UpdateManager#key=
|
|||
}.join(', ')})#{o.alias ? " AS #{visit o.alias}" : ''}"
|
||||
end
|
||||
|
||||
def visit_Arel_Nodes_Extract o
|
||||
"EXTRACT(#{o.field.to_s.upcase} FROM #{visit o.expr})#{o.alias ? " AS #{visit o.alias}" : ''}"
|
||||
end
|
||||
|
||||
def visit_Arel_Nodes_Count o
|
||||
"COUNT(#{o.distinct ? 'DISTINCT ' : ''}#{o.expressions.map { |x|
|
||||
visit x
|
||||
|
|
9
lib/arel/window_predications.rb
Normal file
9
lib/arel/window_predications.rb
Normal file
|
@ -0,0 +1,9 @@
|
|||
module Arel
|
||||
module WindowPredications
|
||||
|
||||
def over(expr = nil)
|
||||
Nodes::Over.new(self, expr)
|
||||
end
|
||||
|
||||
end
|
||||
end
|
19
test/nodes/test_extract.rb
Normal file
19
test/nodes/test_extract.rb
Normal file
|
@ -0,0 +1,19 @@
|
|||
require 'helper'
|
||||
|
||||
describe Arel::Nodes::Extract do
|
||||
it "should extract field" do
|
||||
table = Arel::Table.new :users
|
||||
table[:timestamp].extract('date').to_sql.must_be_like %{
|
||||
EXTRACT(DATE FROM "users"."timestamp")
|
||||
}
|
||||
end
|
||||
|
||||
describe "as" do
|
||||
it 'should alias the extract' do
|
||||
table = Arel::Table.new :users
|
||||
table[:timestamp].extract('date').as('foo').to_sql.must_be_like %{
|
||||
EXTRACT(DATE FROM "users"."timestamp") AS foo
|
||||
}
|
||||
end
|
||||
end
|
||||
end
|
49
test/nodes/test_over.rb
Normal file
49
test/nodes/test_over.rb
Normal file
|
@ -0,0 +1,49 @@
|
|||
require 'helper'
|
||||
|
||||
describe Arel::Nodes::Over do
|
||||
describe 'as' do
|
||||
it 'should alias the expression' do
|
||||
table = Arel::Table.new :users
|
||||
table[:id].count.over.as('foo').to_sql.must_be_like %{
|
||||
COUNT("users"."id") OVER () AS foo
|
||||
}
|
||||
end
|
||||
end
|
||||
|
||||
describe 'with literal' do
|
||||
it 'should reference the window definition by name' do
|
||||
table = Arel::Table.new :users
|
||||
table[:id].count.over('foo').to_sql.must_be_like %{
|
||||
COUNT("users"."id") OVER "foo"
|
||||
}
|
||||
end
|
||||
end
|
||||
|
||||
describe 'with SQL literal' do
|
||||
it 'should reference the window definition by name' do
|
||||
table = Arel::Table.new :users
|
||||
table[:id].count.over(Arel.sql('foo')).to_sql.must_be_like %{
|
||||
COUNT("users"."id") OVER foo
|
||||
}
|
||||
end
|
||||
end
|
||||
|
||||
describe 'with no expression' do
|
||||
it 'should use empty definition' do
|
||||
table = Arel::Table.new :users
|
||||
table[:id].count.over.to_sql.must_be_like %{
|
||||
COUNT("users"."id") OVER ()
|
||||
}
|
||||
end
|
||||
end
|
||||
|
||||
describe 'with expression' do
|
||||
it 'should use definition in sub-expression' do
|
||||
table = Arel::Table.new :users
|
||||
window = Arel::Nodes::Window.new.order(table['foo'])
|
||||
table[:id].count.over(window).to_sql.must_be_like %{
|
||||
COUNT("users"."id") OVER (ORDER BY \"users\".\"foo\")
|
||||
}
|
||||
end
|
||||
end
|
||||
end
|
|
@ -731,6 +731,162 @@ module Arel
|
|||
end
|
||||
end
|
||||
|
||||
describe 'window definition' do
|
||||
it 'can be empty' do
|
||||
table = Table.new :users
|
||||
manager = Arel::SelectManager.new Table.engine
|
||||
manager.from table
|
||||
manager.window('a_window')
|
||||
manager.to_sql.must_be_like %{
|
||||
SELECT FROM "users" WINDOW "a_window" AS ()
|
||||
}
|
||||
end
|
||||
|
||||
it 'takes an order' do
|
||||
table = Table.new :users
|
||||
manager = Arel::SelectManager.new Table.engine
|
||||
manager.from table
|
||||
manager.window('a_window').order(table['foo'].asc)
|
||||
manager.to_sql.must_be_like %{
|
||||
SELECT FROM "users" WINDOW "a_window" AS (ORDER BY "users"."foo" ASC)
|
||||
}
|
||||
end
|
||||
|
||||
it 'takes a rows frame, unbounded preceding' do
|
||||
table = Table.new :users
|
||||
manager = Arel::SelectManager.new Table.engine
|
||||
manager.from table
|
||||
manager.window('a_window').rows(Arel::Nodes::Preceding.new)
|
||||
manager.to_sql.must_be_like %{
|
||||
SELECT FROM "users" WINDOW "a_window" AS (ROWS UNBOUNDED PRECEDING)
|
||||
}
|
||||
end
|
||||
|
||||
it 'takes a rows frame, bounded preceding' do
|
||||
table = Table.new :users
|
||||
manager = Arel::SelectManager.new Table.engine
|
||||
manager.from table
|
||||
manager.window('a_window').rows(Arel::Nodes::Preceding.new(5))
|
||||
manager.to_sql.must_be_like %{
|
||||
SELECT FROM "users" WINDOW "a_window" AS (ROWS 5 PRECEDING)
|
||||
}
|
||||
end
|
||||
|
||||
it 'takes a rows frame, unbounded following' do
|
||||
table = Table.new :users
|
||||
manager = Arel::SelectManager.new Table.engine
|
||||
manager.from table
|
||||
manager.window('a_window').rows(Arel::Nodes::Following.new)
|
||||
manager.to_sql.must_be_like %{
|
||||
SELECT FROM "users" WINDOW "a_window" AS (ROWS UNBOUNDED FOLLOWING)
|
||||
}
|
||||
end
|
||||
|
||||
it 'takes a rows frame, bounded following' do
|
||||
table = Table.new :users
|
||||
manager = Arel::SelectManager.new Table.engine
|
||||
manager.from table
|
||||
manager.window('a_window').rows(Arel::Nodes::Following.new(5))
|
||||
manager.to_sql.must_be_like %{
|
||||
SELECT FROM "users" WINDOW "a_window" AS (ROWS 5 FOLLOWING)
|
||||
}
|
||||
end
|
||||
|
||||
it 'takes a rows frame, current row' do
|
||||
table = Table.new :users
|
||||
manager = Arel::SelectManager.new Table.engine
|
||||
manager.from table
|
||||
manager.window('a_window').rows(Arel::Nodes::CurrentRow.new)
|
||||
manager.to_sql.must_be_like %{
|
||||
SELECT FROM "users" WINDOW "a_window" AS (ROWS CURRENT ROW)
|
||||
}
|
||||
end
|
||||
|
||||
it 'takes a rows frame, between two delimiters' do
|
||||
table = Table.new :users
|
||||
manager = Arel::SelectManager.new Table.engine
|
||||
manager.from table
|
||||
window = manager.window('a_window')
|
||||
window.frame(
|
||||
Arel::Nodes::Between.new(
|
||||
window.rows,
|
||||
Nodes::And.new([
|
||||
Arel::Nodes::Preceding.new,
|
||||
Arel::Nodes::CurrentRow.new
|
||||
])))
|
||||
manager.to_sql.must_be_like %{
|
||||
SELECT FROM "users" WINDOW "a_window" AS (ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW)
|
||||
}
|
||||
end
|
||||
|
||||
it 'takes a range frame, unbounded preceding' do
|
||||
table = Table.new :users
|
||||
manager = Arel::SelectManager.new Table.engine
|
||||
manager.from table
|
||||
manager.window('a_window').range(Arel::Nodes::Preceding.new)
|
||||
manager.to_sql.must_be_like %{
|
||||
SELECT FROM "users" WINDOW "a_window" AS (RANGE UNBOUNDED PRECEDING)
|
||||
}
|
||||
end
|
||||
|
||||
it 'takes a range frame, bounded preceding' do
|
||||
table = Table.new :users
|
||||
manager = Arel::SelectManager.new Table.engine
|
||||
manager.from table
|
||||
manager.window('a_window').range(Arel::Nodes::Preceding.new(5))
|
||||
manager.to_sql.must_be_like %{
|
||||
SELECT FROM "users" WINDOW "a_window" AS (RANGE 5 PRECEDING)
|
||||
}
|
||||
end
|
||||
|
||||
it 'takes a range frame, unbounded following' do
|
||||
table = Table.new :users
|
||||
manager = Arel::SelectManager.new Table.engine
|
||||
manager.from table
|
||||
manager.window('a_window').range(Arel::Nodes::Following.new)
|
||||
manager.to_sql.must_be_like %{
|
||||
SELECT FROM "users" WINDOW "a_window" AS (RANGE UNBOUNDED FOLLOWING)
|
||||
}
|
||||
end
|
||||
|
||||
it 'takes a range frame, bounded following' do
|
||||
table = Table.new :users
|
||||
manager = Arel::SelectManager.new Table.engine
|
||||
manager.from table
|
||||
manager.window('a_window').range(Arel::Nodes::Following.new(5))
|
||||
manager.to_sql.must_be_like %{
|
||||
SELECT FROM "users" WINDOW "a_window" AS (RANGE 5 FOLLOWING)
|
||||
}
|
||||
end
|
||||
|
||||
it 'takes a range frame, current row' do
|
||||
table = Table.new :users
|
||||
manager = Arel::SelectManager.new Table.engine
|
||||
manager.from table
|
||||
manager.window('a_window').range(Arel::Nodes::CurrentRow.new)
|
||||
manager.to_sql.must_be_like %{
|
||||
SELECT FROM "users" WINDOW "a_window" AS (RANGE CURRENT ROW)
|
||||
}
|
||||
end
|
||||
|
||||
it 'takes a range frame, between two delimiters' do
|
||||
table = Table.new :users
|
||||
manager = Arel::SelectManager.new Table.engine
|
||||
manager.from table
|
||||
window = manager.window('a_window')
|
||||
window.frame(
|
||||
Arel::Nodes::Between.new(
|
||||
window.range,
|
||||
Nodes::And.new([
|
||||
Arel::Nodes::Preceding.new,
|
||||
Arel::Nodes::CurrentRow.new
|
||||
])))
|
||||
manager.to_sql.must_be_like %{
|
||||
SELECT FROM "users" WINDOW "a_window" AS (RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW)
|
||||
}
|
||||
end
|
||||
end
|
||||
|
||||
describe 'delete' do
|
||||
it "copies from" do
|
||||
engine = EngineProxy.new Table.engine
|
||||
|
|
|
@ -179,7 +179,8 @@ module Arel
|
|||
core.froms = :b
|
||||
core.wheres << :c
|
||||
core.groups << :d
|
||||
core.having = :e
|
||||
core.windows << :e
|
||||
core.having = :f
|
||||
|
||||
@visitor.accept core
|
||||
assert_equal [
|
||||
|
@ -188,7 +189,8 @@ module Arel
|
|||
core.source,
|
||||
:c, core.wheres,
|
||||
:d, core.groups,
|
||||
:e,
|
||||
:e, core.windows,
|
||||
:f,
|
||||
core], @collector.calls
|
||||
end
|
||||
|
||||
|
|
Loading…
Reference in a new issue