forked from OSAS/community-website
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfetch-dashboard.rb
executable file
·98 lines (78 loc) · 3.09 KB
/
fetch-dashboard.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
#!/usr/bin/ruby
# Dependencies: rubygem-activesupport
require 'bundler'
Bundler.setup
require 'time'
require 'fileutils'
require 'net/http'
require 'uri'
require 'io/console'
require 'json'
require 'yaml'
require 'active_support'
require 'active_support/core_ext'
base_dir = 'data/dashboard/projects'
sites = YAML.load_file('dashboard_config.yml')
# Current terminal width, in characters (used for the progressbar)
screen_width = IO.console.winsize[1] if IO.console
sites.each do |site|
site = site.with_indifferent_access
site_name = site[:name]
uri = URI.parse site[:url]
# Set progressbar counts
pb_total = pb_current = site[:stats].length
# Open the HTTP(s) connection to the server.
# If the server supports it, all of the commands will be performed on
# one connection, saving overhead and making it all quicker.
Net::HTTP.start(uri.host, uri.port, use_ssl: uri.scheme[/s/]) do |h|
prefix = "#{base_dir}/#{site_name.downcase}"
prefix_exist = Dir.exist?(prefix)
if IO.console
# Progressbar settings (per site)
pb_action = prefix_exist ? 'Freshening' : 'Downloading'
pb_str = "#{pb_action} #{site_name} dashboard data"
pb_width = screen_width - pb_str.length - 10
end
# Re-use the site config above by outputting specific metadata to files
FileUtils.mkdir_p prefix
File.write "#{prefix}/details.json", site.to_json
site[:stats].each do |stat|
# Calculate and ouput our progressbar!
if IO.console
pb_current -= 1
pb_done = (pb_total - pb_current).to_f / pb_total.to_f
pb_bar = '#' * (pb_done * pb_width).to_i
pb_empty = '-' * ((1 - pb_done) * pb_width).to_i
print "\r#{pb_str}: #{(pb_done * 100).to_i}% #{pb_bar}#{pb_empty}"
end
%w(static evolutionary).each do |type|
type_short = type.slice(0, 4)
filename = "#{prefix}/#{type_short}/#{stat}.json"
remote_path = "#{uri.path}#{stat}-#{type}.json"
# Skip heads if path doesn't exist
response = prefix_exist ? h.head(remote_path) : nil
# Only operate on found remote files
if !response || response.code.match(/^2/)
# mtimes may result in nil/false; it works fine for comparison below
mtime = response && Time.parse(response.header['last-modified']).utc
mtime_file = File.exist?(filename) && File.mtime(filename).utc
# Download unless it's the same exact time local as remote
unless File.exist?(filename) && mtime_file == mtime
remote = h.get remote_path
# Use GET for mtime if we skipped HEAD
mtime ||= Time.parse(remote.header['last-modified']).utc
# Ensure path exists & write file with proper date
FileUtils.mkdir_p File.dirname(filename)
File.write filename, remote.body
FileUtils.touch filename, mtime: mtime
end
else
code = response ? response.code : remote.code
puts "Got a #{code} when trying to download #{uri.host}#{remote_path}"
end
end
end
puts '' if IO.console
end
end
puts '' if IO.console