-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathgrafanimate.rb
139 lines (121 loc) · 5.02 KB
/
grafanimate.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
require 'active_support'
require 'active_support/core_ext'
require "selenium-webdriver"
module Grafanimate
class Base
#CROP = false
SELENIUM_URL = ENV['SELENIUM_URL']
TIME_FORMAT = '%Y-%m-%d %H:%M'
TIMEOUT = 60
attr_reader :driver, :wait
def initialize(url, from, to)
@from, @to = from, to
logger.info("Starting screenshots for #{from} to #{to}")
client = Selenium::WebDriver::Remote::Http::Default.new
client.read_timeout = 120 # seconds
@driver = Selenium::WebDriver.for :remote, url: SELENIUM_URL, capabilities: :firefox, http_client: client
@wait = Selenium::WebDriver::Wait.new(timeout: TIMEOUT) # seconds
driver.manage.window.resize_to(1200, 1200+85) # <- resizes the window
driver.action.move_to_location(0,0)
@driver.navigate.to(url)
@wait.until { driver.find_element(:css, 'button[data-testid="data-testid TimePicker Open Button"]') }
end
def wait_for_panels
@wait.until { @driver.execute_script("panels = Object.values(window.grafanaRuntime.getPanelData());return panels.length && panels.every(function(o) {return o?.state=='Done'})") }
end
def wait_for_scene_state
@wait.until { @driver.execute_script("return __grafanaSceneContext.state.$data._state.data.state == 'Done'") }
end
def wait_multiple
body = @driver.find_element(:tag_name, 'body')
@wait.until do
tests = []
tests << !body.text.include?('Loading')
#tests << (@driver.execute_script("panels = Object.values(window.grafanaRuntime.getPanelData());return panels.length && panels.every(function(o) {return o?.state=='Done'})") rescue false)<
tests << @driver.execute_script("return __grafanaSceneContext.state.$data._state.data.state == 'Done'")
tests << @driver.execute_script("return __grafanaSceneContext.state.$data._state.data.series.length > 0")
tests << @driver.execute_script("return document.querySelector('[aria-label=\"Refresh\"]')")
tests << @driver.execute_script("return !document.querySelector('[aria-label=\"Cancel\"]')")
break if tests.all?
if tests.any?
logger.warn "Inconsistent tests: #{tests.inspect}"
end
false
end
end
def screenshot(t)
path = "render/picture-#{t.strftime(TIME_FORMAT)}.png"
logger.benchmark_info(path) do
picture = @driver.screenshot_as(:png)
File.binwrite(path, picture)
end
end
def self.render
#video_framerate = 1.5
#video_fps = 30
video_framerate = 5
video_fps = 5
source = 'render/*.png'
target = 'render.mp4'
vf = []
#vf << "framerate=fps=30:interp_start=0:interp_end=255:scene=100"
vf << self::CROP if self::CROP
vf += ["pad=ceil(iw/2)*2:ceil(ih/2)*2", "fps=#{video_fps}", "format=yuv420p"]
c = ['ffmpeg']
c += ['-framerate', video_framerate.to_s]
c += %w(-pattern_type glob)
c += ['-i', source]
c += %w(-c:v libx264)
c += %w(-preset veryslow)
c += %w(-profile high422)
c += %w(-movflags +faststart)
c += ['-vf', vf.join(', ')]
c << target
c << '-y'
#require 'pry' ; binding.pry
system(*c)
end
def quit
@driver.quit
end
def scenesapi
body = @driver.find_element(:tag_name, 'body')
wait_multiple
t = @from
until t >= @to
@driver.execute_script <<-JS
__grafanaSceneContext.state.$timeRange.setState({ from: '#{t.strftime(TIME_FORMAT)}', to: '#{(t+1.hour).strftime(TIME_FORMAT)}' });
__grafanaSceneContext.state.$timeRange.onRefresh();
JS
sleep 0.1
wait_multiple
screenshot(t)
t += 1.hour
end
end
end
class PriceMap < Base
include SemanticLogger::Loggable
URL = "http://grafana.monitoring/d/fa529e06-ff34-415d-adf1-dde1a6f28350/prices-plotly-map?orgId=1&var-region=europe&var-area=All&var-scale_max=300&var-min_interval=5m&var-frame_duration=150&kiosk"
CROP = 'crop=1074:953:104:175'
def initialize
super(URL, 1.day.from_now.beginning_of_day, 2.days.from_now.beginning_of_day - 1.hour)
end
end
class TransmissionMap < Base
include SemanticLogger::Loggable
URL = "http://grafana.monitoring/d/ae4xmqh628a9sf/transmission-plotly-map?orgId=1&var-region=europe&var-area=$__all&var-min_interval=1h"
CROP = 'crop=1074:953:104:226'
def initialize
#super(URL, 7.days.ago.beginning_of_day, Time.now.beginning_of_day - 1.hour)
super(URL, 7.days.ago.beginning_of_day, Time.now.beginning_of_day - 1.hour)
end
end
class NuclearMap < Base
URL = "http://grafana.monitoring/d/adk4o41xfjncwb/generation-of-peak-plotly-map?from=now-30d&to=now&orgId=1&var-region=argentina&var-region=brazil&var-region=canada&var-region=europe&var-region=south_africa&var-region=taiwan&var-region=usa&var-area=All&var-production_type=14&var-min_interval=1h&var-colorscale=Electric"
CROP = 'crop=1074:953:104:52'
def initialize
super(URL, 1.day.ago.beginning_of_day, Date.tomorrow - 1.hour)
end
end
end