-
Notifications
You must be signed in to change notification settings - Fork 18
/
lttb_sinatra.rb
140 lines (117 loc) · 3.26 KB
/
lttb_sinatra.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
# ruby lttb.rb postgres://user:pass@host:port/db_name
require 'bundler/inline' #require only what you need
gemfile(true) do
gem 'timescaledb', path: '../../..'
gem 'pry'
gem 'sinatra', require: false
gem 'sinatra-reloader', require: false
gem 'sinatra-cross_origin', require: false
gem 'chartkick'
gem 'puma'
end
require 'timescaledb/toolkit'
require 'sinatra'
require 'sinatra/json'
require 'sinatra/cross_origin'
require 'chartkick'
require_relative 'lttb'
PG_URI = ARGV.last
VALID_SIZES = %i[small med big]
def download_weather_dataset size: :small
unless VALID_SIZES.include?(size)
fail "Invalid size: #{size}. Valids are #{VALID_SIZES}"
end
url = "https://timescaledata.blob.core.windows.net/datasets/weather_#{size}.tar.gz"
puts "fetching #{size} weather dataset..."
system "wget \"#{url}\""
puts "done!"
end
def setup size: :small
file = "weather_#{size}.tar.gz"
download_weather_dataset(size: size) unless File.exists? file
puts "extracting #{file}"
system "tar -xvzf #{file} "
puts "creating data structures"
system "psql #{PG_URI} < weather.sql"
system %|psql #{PG_URI} -c "\\COPY locations FROM weather_#{size}_locations.csv CSV"|
system %|psql #{PG_URI} -c "\\COPY conditions FROM weather_#{size}_conditions.csv CSV"|
end
ActiveRecord::Base.establish_connection(PG_URI)
class Location < ActiveRecord::Base
self.primary_key = "device_id"
has_many :conditions, foreign_key: "device_id"
end
class Condition < ActiveRecord::Base
acts_as_hypertable time_column: "time"
acts_as_time_vector value_column: "temperature", segment_by: "device_id"
belongs_to :location, foreign_key: "device_id"
end
# Setup Hypertable as in a migration
ActiveRecord::Base.connection.instance_exec do
ActiveRecord::Base.logger = Logger.new(STDOUT)
unless Condition.table_exists?
setup size: :big
end
end
require 'sinatra/reloader'
require 'sinatra/contrib'
register Sinatra::Reloader
register Sinatra::Contrib
include Chartkick::Helper
set :bind, '0.0.0.0'
set :port, 9999
def conditions
device_ids = (1..9).map{|i|"weather-pro-00000#{i}"}
Condition
.where(device_id: device_ids.first)
.order('time')
end
def threshold
params[:threshold]&.to_i || 50
end
configure do
enable :cross_origin
end
before do
response.headers['Access-Control-Allow-Origin'] = '*'
end
# routes...
options "*" do
response.headers["Allow"] = "GET, PUT, POST, DELETE, OPTIONS"
response.headers["Access-Control-Allow-Headers"] = "Authorization,
Content-Type, Accept, X-User-Email, X-Auth-Token"
response.headers["Access-Control-Allow-Origin"] = "*"
200
end
get '/' do
headers 'Access-Control-Allow-Origin' => 'https://cdn.jsdelivr.net/'
erb :index
end
get '/lttb_ruby' do
payload = conditions
.pluck(:device_id, :time, :temperature)
.group_by(&:first)
.map do |device_id, data|
data.each(&:shift)
{
name: device_id,
data: Lttb.downsample(data, threshold)
}
end
json payload
end
get "/lttb_sql" do
downsampled = conditions
.lttb(threshold: threshold)
.map do |device_id, data|
{
name: device_id,
data: data.sort_by(&:first)
}
end
json downsampled
end
get '/all_data' do
data = conditions.pluck(:time, :temperature)
json [ { name: "All data", data: data} ]
end