prosperity 0.0.1 → 0.0.2
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/app/assets/javascripts/prosperity/application.js.coffee +5 -24
- data/app/assets/javascripts/prosperity/dashboards.js.coffee +17 -0
- data/app/assets/javascripts/prosperity/graph.js.coffee +89 -0
- data/app/assets/stylesheets/prosperity/application.css.scss +1 -1
- data/app/assets/stylesheets/prosperity/dashboards.css +4 -0
- data/app/assets/stylesheets/prosperity/graph.css +4 -0
- data/app/controllers/prosperity/application_controller.rb +30 -0
- data/app/controllers/prosperity/dashboard_graphs_controller.rb +24 -0
- data/app/controllers/prosperity/dashboards_controller.rb +33 -0
- data/app/controllers/prosperity/graphs_controller.rb +72 -0
- data/app/controllers/prosperity/metrics_controller.rb +57 -0
- data/app/helpers/prosperity/application_helper.rb +4 -0
- data/app/helpers/prosperity/dashboard_graphs_helper.rb +4 -0
- data/app/helpers/prosperity/dashboards_helper.rb +4 -0
- data/app/helpers/prosperity/graph_helper.rb +7 -0
- data/app/models/prosperity/dashboard.rb +8 -0
- data/app/models/prosperity/dashboard_graph.rb +6 -0
- data/app/models/prosperity/graph.rb +20 -0
- data/app/models/prosperity/graph_line.rb +8 -0
- data/app/views/layouts/prosperity/application.html.erb +56 -0
- data/app/views/prosperity/dashboards/edit.html.erb +24 -0
- data/app/views/prosperity/dashboards/index.html.erb +15 -0
- data/app/views/prosperity/dashboards/new.html.erb +8 -0
- data/app/views/prosperity/dashboards/show.html.erb +14 -0
- data/app/views/prosperity/graphs/edit.html.erb +41 -0
- data/app/views/prosperity/graphs/new.html.erb +15 -0
- data/app/views/prosperity/metrics/index.html.erb +8 -0
- data/app/views/prosperity/metrics/show.html.erb +23 -0
- data/app/views/prosperity/shared/_date_range.html.erb +9 -0
- data/config/routes.rb +12 -2
- data/db/migrate/20131127042251_dashboards.rb +34 -0
- data/lib/generators/metric/USAGE +8 -0
- data/lib/generators/metric/metric_generator.rb +7 -0
- data/lib/generators/metric/templates/metric.rb.erb +3 -0
- data/lib/prosperity/aggregate/aggregate_builder.rb +42 -0
- data/lib/prosperity/aggregate/average.rb +12 -0
- data/lib/prosperity/aggregate/base.rb +5 -0
- data/lib/prosperity/aggregate/count.rb +12 -0
- data/lib/prosperity/aggregate/maximum.rb +12 -0
- data/lib/prosperity/aggregate/minimum.rb +12 -0
- data/lib/prosperity/aggregate/sql.rb +12 -0
- data/lib/prosperity/aggregate/sum.rb +12 -0
- data/lib/prosperity/aggregate/with_column.rb +10 -0
- data/lib/prosperity/aggregate.rb +5 -0
- data/lib/prosperity/engine.rb +14 -0
- data/lib/prosperity/exception.rb +19 -1
- data/lib/prosperity/extractors/base.rb +33 -5
- data/lib/prosperity/extractors/change.rb +35 -0
- data/lib/prosperity/extractors/interval.rb +51 -0
- data/lib/prosperity/extractors/total.rb +23 -0
- data/lib/prosperity/helpers/time.rb +12 -0
- data/lib/prosperity/metric.rb +90 -5
- data/lib/prosperity/metric_finder.rb +10 -0
- data/lib/prosperity/metrics/option.rb +2 -0
- data/lib/prosperity/period.rb +4 -3
- data/lib/prosperity/periods.rb +13 -2
- data/lib/prosperity/version.rb +1 -1
- data/spec/controllers/prosperity/dashboard_graphs_controller_spec.rb +45 -0
- data/spec/controllers/prosperity/dashboards_controller_spec.rb +54 -0
- data/spec/controllers/prosperity/graphs_controller_spec.rb +145 -0
- data/spec/controllers/prosperity/metrics_controller_spec.rb +78 -0
- data/spec/dummy/app/models/user.rb +1 -0
- data/spec/dummy/app/prosperity/user_value_sum_metric.rb +4 -0
- data/spec/dummy/app/prosperity/user_value_sum_sql_metric.rb +4 -0
- data/spec/dummy/app/prosperity/users_metric.rb +8 -0
- data/spec/dummy/app/prosperity/users_sql_metric.rb +4 -0
- data/spec/dummy/config/database.yml +0 -8
- data/spec/dummy/config/routes.rb +1 -1
- data/spec/dummy/db/migrate/20140217005117_add_value_to_users.rb +5 -0
- data/spec/dummy/db/schema.rb +37 -1
- data/spec/dummy/db/seeds.rb +3 -1
- data/spec/dummy/log/development.log +76950 -0
- data/spec/dummy/log/test.log +111316 -0
- data/spec/dummy/tmp/cache/assets/development/sass/34a577735054231563e7022ea73e1468db9203ba/application.css.scssc +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/07e8fb2db7d85e29590a6b05160ca825 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/1533bcf4a8f5a745dfda80cebd88d217 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/1643e7f9f85637df140a850af6674f10 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/197492ed379339e17a0f5d01d3b01b8c +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/1c55cf24465f311353197ce336df0178 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/210050da208fb75a75b701bfa4e8470f +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/2733033a48e2e8a6fdd6e4ab371689ee +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/27a59e08207d3ae723f2b2b3d22264c3 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/3e21790900d62ae3954585f08a1cb28a +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/3f76425644a701b85db3d385f077052d +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/448c1e44da04c20f27eaa31ad8c4391e +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/47df4fdca8a2ff6a43c18d7aa1ffd9c5 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/566fbede2036fd06a20e7e52e30ddc7d +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/5f1aaf22720701db690538726e896a39 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/647038c58f26163a9217646e1e5f09ae +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/6882b260ae69f1594eff540d803e5ac3 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/69f4b83d363269ee4122bbbaaa1325d1 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/6e1aee34907ba1ccdbfca2deb18a5adb +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/72c0d4a0eeedae5e79ae49abb24f6d32 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/828a046f0e7dc2dad0eecfe49d31e14d +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/91ae0bd43f98b4f1f111e0bdc178302c +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/97d99b466bedb69706f4815fa26b69f3 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/990cf740a7968767a327f6186313badb +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/a2b14d5c46db32da9183354fb3fcd0bf +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/a6d6196cfd275dea0718553a95f997a5 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/bb1f5ffb0ccbeec2c6499044b356f7c6 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/bc83340720a76d8208112f6aecbff08e +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/be67cef79122a71f7432c2f984f1f258 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/c83e71ece967e80f8df838874c130431 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/cad26971ff72bbf5aa5f4e6a469389eb +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/d0d5517256edcd1f83b11a13c6ca040f +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/d0f0c5e105bfb1d6e5cd7295fbd1f65d +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/d3087f7df919b7f0d8142ee163db6c79 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/decc27a1fa9ea32fe0dcc18619e32587 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/df796f7ea525be06a82ebc3a4f97ffec +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/e02b5049cf477b43a72b4a00e2fceb46 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/e2a9bdd87a2d19256837fce6544a122c +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/e378dcebf3c529db0d6e57d0c8dc3904 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/ee3031bd54230ddf2addb5fa5718c86b +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/f4648036e284d9aa3468f4f7b9bbb967 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/f9bbf698d099ead559e5ba4a5a4c5538 +0 -0
- data/spec/dummy/tmp/cache/assets/development/sprockets/fb8d3825617fcc11dfa614a51effee0e +0 -0
- data/spec/dummy/tmp/cache/assets/test/sass/34a577735054231563e7022ea73e1468db9203ba/application.css.scssc +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/07e8fb2db7d85e29590a6b05160ca825 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/0a40fa5e0d6c0de40a8a698dcd697305 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/1533bcf4a8f5a745dfda80cebd88d217 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/1643e7f9f85637df140a850af6674f10 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/177ca1f3e92db4f8a8f6e7ad9786ebfc +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/197492ed379339e17a0f5d01d3b01b8c +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/1c55cf24465f311353197ce336df0178 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/1cb94582c835e73b3fd3c276f9a7de56 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/203047a4141cf08162077e7290421bbb +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/210050da208fb75a75b701bfa4e8470f +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/2733033a48e2e8a6fdd6e4ab371689ee +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/27a59e08207d3ae723f2b2b3d22264c3 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/29a6c93aeb91d07a5533b3adba7697d0 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/29ac2be474969468157245a7f8737fa4 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/314e44d29ecc7449c9ca3b754329c245 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/3162d1398d6940ee5e933b7ab4d2273d +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/3d80e2a5faaf2bd987d804c17aead745 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/4155a156cc6ffb880b56dbc34346fbc0 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/4166fe47cddf4ba18c53b843d69b4c5f +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/441f922b26aae90d40e319b6ed0a3bc8 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/4432a4b71ce9551df8d4b3860f4b9be3 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/448c1e44da04c20f27eaa31ad8c4391e +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/47df4fdca8a2ff6a43c18d7aa1ffd9c5 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/5479c7b4fba9179d4d1dc59ae64964c1 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/566fbede2036fd06a20e7e52e30ddc7d +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/5a3ab9afd151c265cf11c621c0ddbe00 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/647038c58f26163a9217646e1e5f09ae +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/64e0150fafbd9aaa3822efa1abb0211d +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/65c2d908dbca1bc833ae0b552f6c27fa +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/6882b260ae69f1594eff540d803e5ac3 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/69f4b83d363269ee4122bbbaaa1325d1 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/6e1aee34907ba1ccdbfca2deb18a5adb +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/726d3f6850549a59954f0bf5584cfd9f +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/7ab9ac7cc2529219a4c68fd7854d5c0f +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/7bac3aca131c870ad5b07d8ccdc86e06 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/80392355fc6e00cf69b2cd46978ebcbe +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/874b42463fefbb0bce47775722ebb4ae +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/87ef8ddd2847d19f22a16d726b88b433 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/91ae0bd43f98b4f1f111e0bdc178302c +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/97d99b466bedb69706f4815fa26b69f3 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/990cf740a7968767a327f6186313badb +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/9a690664139b0393787d6454b4697183 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/a2b14d5c46db32da9183354fb3fcd0bf +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/a6d6196cfd275dea0718553a95f997a5 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/bc83340720a76d8208112f6aecbff08e +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/c29c465933e7e86d6f6224f5e78c5baf +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/c975834a35907ecd2d314bbd6844ee88 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/cad26971ff72bbf5aa5f4e6a469389eb +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/cf4affc50c6303e717d6514380d1f417 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/d0d5517256edcd1f83b11a13c6ca040f +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/d3087f7df919b7f0d8142ee163db6c79 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/d8b357d5741711da2dd710af0e15d268 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/decc27a1fa9ea32fe0dcc18619e32587 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/df796f7ea525be06a82ebc3a4f97ffec +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/e22a1b318499e7f06ea9274e18ab004f +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/e298cc3a8203ab1fb6978a90254f8926 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/e378dcebf3c529db0d6e57d0c8dc3904 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/e868f358fe87542c483951a2d9f532b6 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/e8a6adf580ebd1942f4aabe6738678c2 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/ed71f99e0a9653296abd5d6f501dd898 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/efe6f2d16d3a1819817efb8734daf8af +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/f03e99bdb4159ae3485ca2858716e19b +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/f4648036e284d9aa3468f4f7b9bbb967 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/f961f7434a28c0fe24bf4ffe6cba75fb +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/f9bbf698d099ead559e5ba4a5a4c5538 +0 -0
- data/spec/dummy/tmp/cache/assets/test/sprockets/fb8d3825617fcc11dfa614a51effee0e +0 -0
- data/spec/helpers/prosperity/dashboard_graphs_helper_spec.rb +6 -0
- data/spec/helpers/prosperity/dashboards_helper_spec.rb +6 -0
- data/spec/helpers/prosperity/{metrics_helper_spec.rb → graph_helper_spec.rb} +1 -1
- data/spec/lib/prosperity/aggregate/aggregate_builder_spec.rb +65 -0
- data/spec/lib/prosperity/aggregate/average_spec.rb +7 -0
- data/spec/lib/prosperity/aggregate/base_spec.rb +7 -0
- data/spec/lib/prosperity/aggregate/count_spec.rb +7 -0
- data/spec/lib/prosperity/aggregate/maximum_spec.rb +7 -0
- data/spec/lib/prosperity/aggregate/minimum_spec.rb +7 -0
- data/spec/lib/prosperity/aggregate/sql_spec.rb +7 -0
- data/spec/lib/prosperity/aggregate/sum_spec.rb +7 -0
- data/spec/lib/prosperity/aggregate/with_column_spec.rb +7 -0
- data/spec/lib/prosperity/aggregate_spec.rb +7 -0
- data/spec/lib/prosperity/extractors/base_spec.rb +27 -2
- data/spec/lib/prosperity/extractors/change_spec.rb +72 -0
- data/spec/lib/prosperity/extractors/interval_spec.rb +153 -0
- data/spec/lib/prosperity/extractors/total_spec.rb +79 -0
- data/spec/lib/prosperity/helpers/time_spec.rb +7 -0
- data/spec/lib/prosperity/metric_finder_spec.rb +7 -0
- data/spec/lib/prosperity/metric_spec.rb +126 -1
- data/spec/lib/prosperity/periods_spec.rb +14 -0
- data/spec/models/prosperity/dashboard_graph_spec.rb +6 -0
- data/spec/models/prosperity/dashboard_spec.rb +6 -0
- data/spec/models/prosperity/graph_line_spec.rb +6 -0
- data/spec/models/prosperity/graph_spec.rb +6 -0
- data/spec/spec_helper.rb +5 -2
- data/spec/support/shared/extractors.rb +64 -0
- data/spec/support/test_database.rb +1 -0
- data/vendor/assets/javascripts/highcharts.js +285 -0
- metadata +317 -62
- data/app/helpers/prosperity/metrics_helper.rb +0 -11
- data/app/views/layouts/prosperity/application.html.haml +0 -37
- data/app/views/prosperity/metrics/index.html.haml +0 -7
- data/db/migrate/20131026214127_test.rb +0 -4
- data/lib/prosperity/extractors/count.rb +0 -18
- data/lib/prosperity/extractors/group.rb +0 -23
- data/spec/lib/prosperity/extractors/count_spec.rb +0 -35
- data/spec/lib/prosperity/extractors/group_spec.rb +0 -34
- data/vendor/assets/javascripts/chart.js +0 -1426
@@ -0,0 +1,153 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
module Prosperity
|
4
|
+
describe Extractors::Interval do
|
5
|
+
let(:expected_data_size) { 14 }
|
6
|
+
it_behaves_like "an extractor"
|
7
|
+
|
8
|
+
let(:start_time) { 1.year.ago }
|
9
|
+
let(:end_time) { start_time + 1.year }
|
10
|
+
let(:period) { Periods::MONTH }
|
11
|
+
let(:data) { subject.to_a }
|
12
|
+
|
13
|
+
subject { Extractors::Interval.new(metric, 'default', start_time, end_time, period) }
|
14
|
+
|
15
|
+
before do
|
16
|
+
User.delete_all
|
17
|
+
[2.years.ago, 1.month.ago, 1.month.from_now].each do |time|
|
18
|
+
User.create created_at: time
|
19
|
+
end
|
20
|
+
User.create created_at: 1.week.ago
|
21
|
+
end
|
22
|
+
|
23
|
+
context "simple scope" do
|
24
|
+
let(:metric) { UsersMetric.new }
|
25
|
+
|
26
|
+
describe "#to_a" do
|
27
|
+
it "returns the one entry per period" do
|
28
|
+
data.size.should == expected_data_size
|
29
|
+
end
|
30
|
+
|
31
|
+
it "only returns models from that period" do
|
32
|
+
data.sum.should == 2
|
33
|
+
end
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
context "simple sql fragment" do
|
38
|
+
let(:metric) { UsersSqlMetric.new }
|
39
|
+
|
40
|
+
describe "#to_a" do
|
41
|
+
it "returns the one entry per period" do
|
42
|
+
data.size.should == expected_data_size
|
43
|
+
end
|
44
|
+
|
45
|
+
it "only returns models from that period" do
|
46
|
+
data.sum.should == 2
|
47
|
+
end
|
48
|
+
end
|
49
|
+
|
50
|
+
context "with weekly period" do
|
51
|
+
let(:period) { Periods::WEEK }
|
52
|
+
|
53
|
+
describe "#to_a" do
|
54
|
+
it "returns the one entry per period" do
|
55
|
+
[54, 53].should include(data.size)
|
56
|
+
end
|
57
|
+
|
58
|
+
it "only returns models from that period" do
|
59
|
+
data.sum.should == 2
|
60
|
+
end
|
61
|
+
end
|
62
|
+
|
63
|
+
context "with alternate start and end times" do
|
64
|
+
let(:start_time) { 2.weeks.ago }
|
65
|
+
let(:end_time) { start_time + 2.weeks }
|
66
|
+
|
67
|
+
describe "#to_a" do
|
68
|
+
it "returns the one entry per period" do
|
69
|
+
data.size.should == 4
|
70
|
+
end
|
71
|
+
|
72
|
+
it "only returns models from that period" do
|
73
|
+
data.sum.should == 1
|
74
|
+
end
|
75
|
+
end
|
76
|
+
end
|
77
|
+
end
|
78
|
+
end
|
79
|
+
|
80
|
+
context "sql fragment with nested WITH" do
|
81
|
+
let(:metric) do
|
82
|
+
Class.new(Metric) do
|
83
|
+
sql "WITH all_columns AS (SELECT * FROM users) SELECT name, created_at FROM users"
|
84
|
+
end.new
|
85
|
+
end
|
86
|
+
|
87
|
+
describe "#to_a" do
|
88
|
+
it "returns the one entry per period" do
|
89
|
+
data.size.should == expected_data_size
|
90
|
+
end
|
91
|
+
|
92
|
+
it "only returns models from that period" do
|
93
|
+
data.sum.should == 2
|
94
|
+
end
|
95
|
+
end
|
96
|
+
end
|
97
|
+
|
98
|
+
context "a metric with a sum aggregate" do
|
99
|
+
before do
|
100
|
+
User.create! value: 1
|
101
|
+
User.create! value: 3
|
102
|
+
end
|
103
|
+
|
104
|
+
context "a non sql metric" do
|
105
|
+
let(:metric) do
|
106
|
+
Class.new(Metric) do
|
107
|
+
scope { User }
|
108
|
+
aggregate { sum(:value) }
|
109
|
+
end.new
|
110
|
+
end
|
111
|
+
|
112
|
+
describe "#to_a" do
|
113
|
+
it "returns the one entry per period" do
|
114
|
+
data.size.should == expected_data_size
|
115
|
+
data[-2].should == User.all.sum(:value)
|
116
|
+
end
|
117
|
+
end
|
118
|
+
end
|
119
|
+
|
120
|
+
context "a sql metric" do
|
121
|
+
let(:metric) do
|
122
|
+
Class.new(Metric) do
|
123
|
+
sql "SELECT * FROM users"
|
124
|
+
aggregate { sum(:value) }
|
125
|
+
end.new
|
126
|
+
end
|
127
|
+
|
128
|
+
describe "#to_a" do
|
129
|
+
it "returns the one entry per period" do
|
130
|
+
data.size.should == expected_data_size
|
131
|
+
data[-2].should == User.all.sum(:value)
|
132
|
+
end
|
133
|
+
end
|
134
|
+
end
|
135
|
+
|
136
|
+
context "ruby block" do
|
137
|
+
let(:metric) do
|
138
|
+
Class.new(Metric) do
|
139
|
+
value_at do |time, period, *|
|
140
|
+
10
|
141
|
+
end
|
142
|
+
end.new
|
143
|
+
end
|
144
|
+
|
145
|
+
describe "#to_a" do
|
146
|
+
it "delegates to the ruby block" do
|
147
|
+
data.should == [0] * expected_data_size
|
148
|
+
end
|
149
|
+
end
|
150
|
+
end
|
151
|
+
end
|
152
|
+
end
|
153
|
+
end
|
@@ -0,0 +1,79 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
module Prosperity
|
4
|
+
describe Extractors::Total do
|
5
|
+
it_behaves_like "an extractor"
|
6
|
+
let(:expected_data_size) { 14 }
|
7
|
+
|
8
|
+
let(:start_time) { 1.year.ago }
|
9
|
+
let(:end_time) { start_time + 12.months }
|
10
|
+
let(:period) { Periods::MONTH }
|
11
|
+
|
12
|
+
let(:data) { subject.to_a }
|
13
|
+
let(:metric) { UsersMetric.new }
|
14
|
+
let(:option) { 'default' }
|
15
|
+
|
16
|
+
subject { Extractors::Total.new(metric, option, start_time, end_time, period) }
|
17
|
+
|
18
|
+
before do
|
19
|
+
User.delete_all
|
20
|
+
[2.years.ago, 1.month.ago, 1.month.from_now].each do |time|
|
21
|
+
User.create created_at: time
|
22
|
+
end
|
23
|
+
end
|
24
|
+
|
25
|
+
context "simple scope" do
|
26
|
+
describe "#to_a" do
|
27
|
+
it "returns the one entry per period" do
|
28
|
+
data.size.should == expected_data_size
|
29
|
+
end
|
30
|
+
|
31
|
+
it "returns the counts at it increases" do
|
32
|
+
data[0].should == 1
|
33
|
+
data[-1].should == 2
|
34
|
+
end
|
35
|
+
end
|
36
|
+
end
|
37
|
+
|
38
|
+
context "simple sql fragment" do
|
39
|
+
let(:metric) { UsersSqlMetric.new }
|
40
|
+
|
41
|
+
describe "#to_a" do
|
42
|
+
it "returns the one entry per period" do
|
43
|
+
data.size.should == expected_data_size
|
44
|
+
end
|
45
|
+
|
46
|
+
it "returns the counts at it increases" do
|
47
|
+
data[0].should == 1
|
48
|
+
data[-1].should == 2
|
49
|
+
end
|
50
|
+
end
|
51
|
+
end
|
52
|
+
|
53
|
+
context "with an option" do
|
54
|
+
let(:option) { "no_results" }
|
55
|
+
|
56
|
+
describe "#to_a" do
|
57
|
+
it "only returns the results for that option block" do
|
58
|
+
data.all?(&:zero?).should be_true
|
59
|
+
end
|
60
|
+
end
|
61
|
+
end
|
62
|
+
|
63
|
+
context "ruby block" do
|
64
|
+
let(:metric) do
|
65
|
+
Class.new(Metric) do
|
66
|
+
value_at do |time, period, *|
|
67
|
+
10
|
68
|
+
end
|
69
|
+
end.new
|
70
|
+
end
|
71
|
+
|
72
|
+
describe "#to_a" do
|
73
|
+
it "delegates to the ruby block" do
|
74
|
+
data.should == [10] * expected_data_size
|
75
|
+
end
|
76
|
+
end
|
77
|
+
end
|
78
|
+
end
|
79
|
+
end
|
@@ -16,5 +16,12 @@ module Prosperity
|
|
16
16
|
metrics.first.superclass.should == Metric
|
17
17
|
end
|
18
18
|
end
|
19
|
+
|
20
|
+
describe ".find_by_name" do
|
21
|
+
it "finds a metric by name" do
|
22
|
+
metric = described_class.find_by_name('UsersMetric')
|
23
|
+
metric.should be < Metric
|
24
|
+
end
|
25
|
+
end
|
19
26
|
end
|
20
27
|
end
|
@@ -16,6 +16,72 @@ module Prosperity
|
|
16
16
|
it "gets the scope" do
|
17
17
|
subject.scope.should == User
|
18
18
|
end
|
19
|
+
|
20
|
+
it "should not be sql" do
|
21
|
+
subject.should_not be_sql
|
22
|
+
end
|
23
|
+
|
24
|
+
it "should have the default aggregate" do
|
25
|
+
subject.aggregate.should be_an(Aggregate::Count)
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
context "A raw sql metric" do
|
30
|
+
context "From a string" do
|
31
|
+
subject do
|
32
|
+
Class.new(Metric) do
|
33
|
+
sql "SELECT name, created_at FROM users"
|
34
|
+
end.new
|
35
|
+
end
|
36
|
+
|
37
|
+
it "groups by created_at by default" do
|
38
|
+
subject.group_by.should == :created_at
|
39
|
+
end
|
40
|
+
|
41
|
+
it "gets the scope" do
|
42
|
+
subject.sql.should == "SELECT name, created_at FROM users"
|
43
|
+
end
|
44
|
+
|
45
|
+
it "raises an exception when adding an option" do
|
46
|
+
expect {
|
47
|
+
subject.class.option "active" do |scope|
|
48
|
+
scope.active
|
49
|
+
end
|
50
|
+
}.to raise_exception(SqlMetricCannotHaveOption)
|
51
|
+
end
|
52
|
+
|
53
|
+
it "should be sql" do
|
54
|
+
subject.should be_sql
|
55
|
+
end
|
56
|
+
end
|
57
|
+
|
58
|
+
context "From a block" do
|
59
|
+
subject do
|
60
|
+
Class.new(Metric) do
|
61
|
+
sql { "SELECT name, created_at FROM users" }
|
62
|
+
end.new
|
63
|
+
end
|
64
|
+
|
65
|
+
it "groups by created_at by default" do
|
66
|
+
subject.group_by.should == :created_at
|
67
|
+
end
|
68
|
+
|
69
|
+
it "gets the scope" do
|
70
|
+
subject.sql.should == "SELECT name, created_at FROM users"
|
71
|
+
end
|
72
|
+
|
73
|
+
it "raises an exception when adding an option" do
|
74
|
+
expect {
|
75
|
+
subject.class.option "active" do |scope|
|
76
|
+
scope.active
|
77
|
+
end
|
78
|
+
}.to raise_exception(SqlMetricCannotHaveOption)
|
79
|
+
end
|
80
|
+
|
81
|
+
it "should be sql" do
|
82
|
+
subject.should be_sql
|
83
|
+
end
|
84
|
+
end
|
19
85
|
end
|
20
86
|
|
21
87
|
context "a metric missing the scope" do
|
@@ -29,11 +95,22 @@ module Prosperity
|
|
29
95
|
end
|
30
96
|
end
|
31
97
|
|
98
|
+
context "a metric missing the sql" do
|
99
|
+
subject do
|
100
|
+
Class.new(Metric) do
|
101
|
+
end
|
102
|
+
end
|
103
|
+
|
104
|
+
it "raises an exception when accessing the sql" do
|
105
|
+
expect { subject.sql }.to raise_exception(MissingSql)
|
106
|
+
end
|
107
|
+
end
|
108
|
+
|
32
109
|
context "A simple scope metric with multiple options" do
|
33
110
|
subject do
|
34
111
|
Class.new(Metric) do
|
35
112
|
scope { User }
|
36
|
-
|
113
|
+
option "active" do |scope|
|
37
114
|
scope.active
|
38
115
|
end
|
39
116
|
end.new
|
@@ -45,5 +122,53 @@ module Prosperity
|
|
45
122
|
subject.options['active'].should be_present
|
46
123
|
end
|
47
124
|
end
|
125
|
+
|
126
|
+
context "A metric with a custom group by" do
|
127
|
+
subject do
|
128
|
+
Class.new(Metric) do
|
129
|
+
scope { User }
|
130
|
+
group_by "users.created_at"
|
131
|
+
end.new
|
132
|
+
end
|
133
|
+
|
134
|
+
its(:group_by) { should == 'users.created_at' }
|
135
|
+
end
|
136
|
+
|
137
|
+
context "A metric with a a sum aggregate" do
|
138
|
+
subject do
|
139
|
+
Class.new(Metric) do
|
140
|
+
scope { User }
|
141
|
+
aggregate { sum(:some_column) }
|
142
|
+
end.new
|
143
|
+
end
|
144
|
+
|
145
|
+
let(:aggregate) { subject.aggregate }
|
146
|
+
|
147
|
+
it "has the correct aggregate info" do
|
148
|
+
aggregate.should be_an(Aggregate::Sum)
|
149
|
+
aggregate.column.should == :some_column
|
150
|
+
end
|
151
|
+
end
|
152
|
+
|
153
|
+
context "a ruby code metric" do
|
154
|
+
subject do
|
155
|
+
Class.new(Metric) do
|
156
|
+
value_at do |time, period, option|
|
157
|
+
:expected
|
158
|
+
end
|
159
|
+
end.new
|
160
|
+
end
|
161
|
+
|
162
|
+
it "has the correct value_at info" do
|
163
|
+
subject.value_at.call.should == :expected
|
164
|
+
end
|
165
|
+
|
166
|
+
describe ".ruby?" do
|
167
|
+
it "should be true" do
|
168
|
+
subject.class.ruby?.should == true
|
169
|
+
subject.ruby?.should == true
|
170
|
+
end
|
171
|
+
end
|
172
|
+
end
|
48
173
|
end
|
49
174
|
end
|
@@ -2,6 +2,20 @@ require 'spec_helper'
|
|
2
2
|
|
3
3
|
module Prosperity
|
4
4
|
describe Periods do
|
5
|
+
subject { Periods::DAY }
|
5
6
|
|
7
|
+
describe "#each_period" do
|
8
|
+
it "iterates until the end time" do
|
9
|
+
times = []
|
10
|
+
start_time = DateTime.new(2012, 01, 01)
|
11
|
+
end_time = DateTime.new(2012, 01, 05)
|
12
|
+
|
13
|
+
subject.each_period(start_time, end_time) do |start_time|
|
14
|
+
times << start_time
|
15
|
+
end
|
16
|
+
|
17
|
+
times.should == Array.new(6) { |i| DateTime.new(2012, 01, i + 1) }
|
18
|
+
end
|
19
|
+
end
|
6
20
|
end
|
7
21
|
end
|
data/spec/spec_helper.rb
CHANGED
@@ -5,10 +5,13 @@ require 'rspec/rails'
|
|
5
5
|
require 'rspec/autorun'
|
6
6
|
require 'pry-debugger'
|
7
7
|
|
8
|
-
|
9
|
-
|
8
|
+
PROSPERITY_ROOT = File.expand_path("..", __FILE__)
|
9
|
+
|
10
|
+
Dir[File.join(PROSPERITY_ROOT, "support/**/*.rb")].each {|f| require f }
|
11
|
+
TEST_FILES = File.join(PROSPERITY_ROOT, "test_files")
|
10
12
|
|
11
13
|
RSpec.configure do |config|
|
14
|
+
config.render_views
|
12
15
|
config.treat_symbols_as_metadata_keys_with_true_values = true
|
13
16
|
config.run_all_when_everything_filtered = true
|
14
17
|
config.filter_run :focus
|
@@ -0,0 +1,64 @@
|
|
1
|
+
shared_examples "an extractor" do
|
2
|
+
AGGREGATES = [:sum, :minimum, :maximum, :average]
|
3
|
+
let(:expected_data_size) { 14 }
|
4
|
+
|
5
|
+
["sql", "normal"].each do |type|
|
6
|
+
AGGREGATES.each do |agg|
|
7
|
+
context "a #{type} metric with aggregate #{agg}" do
|
8
|
+
let(:metric) do
|
9
|
+
Class.new(Prosperity::Metric) do
|
10
|
+
if type == 'sql'
|
11
|
+
sql "SELECT * FROM users"
|
12
|
+
else
|
13
|
+
scope { User }
|
14
|
+
end
|
15
|
+
aggregate { send(agg, :value) }
|
16
|
+
end.new
|
17
|
+
end
|
18
|
+
|
19
|
+
describe "#to_a" do
|
20
|
+
let(:data) { subject.to_a }
|
21
|
+
it "returns the one entry per period" do
|
22
|
+
data.size.should == expected_data_size
|
23
|
+
end
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
|
28
|
+
context "a sql metric with default aggregat" do
|
29
|
+
let(:metric) do
|
30
|
+
Class.new(Prosperity::Metric) do
|
31
|
+
if type == 'sql'
|
32
|
+
sql "SELECT * FROM users"
|
33
|
+
else
|
34
|
+
scope { User }
|
35
|
+
end
|
36
|
+
end.new
|
37
|
+
end
|
38
|
+
|
39
|
+
describe "#to_a" do
|
40
|
+
let(:data) { subject.to_a }
|
41
|
+
it "returns the one entry per period" do
|
42
|
+
data.size.should == expected_data_size
|
43
|
+
end
|
44
|
+
end
|
45
|
+
end
|
46
|
+
end
|
47
|
+
|
48
|
+
context "a full SQL metric" do
|
49
|
+
let(:metric) do
|
50
|
+
Class.new(Prosperity::Metric) do
|
51
|
+
sql "SELECT * FROM users"
|
52
|
+
group_by "created_at"
|
53
|
+
aggregate { "SUM(value)" }
|
54
|
+
end.new
|
55
|
+
end
|
56
|
+
|
57
|
+
describe "#to_a" do
|
58
|
+
let(:data) { subject.to_a }
|
59
|
+
it "returns the one entry per period" do
|
60
|
+
data.size.should == expected_data_size
|
61
|
+
end
|
62
|
+
end
|
63
|
+
end
|
64
|
+
end
|