6d7c344b976104b8d56d4e98b06b9dbac5c5a1f0
[packages.grml.org.git] / update.rb
1 #!/usr/bin/env ruby
2 require 'git'
3 require 'erb'
4 require 'yaml'
5 require 'net/http'
6 require 'uri'
7 require 'zlib'
8 require 'yaml'
9
10 DEBUG = false
11
12 def build_used_package_list(source_uri)
13   source_uri.map do |source_uri|
14     fetch_file(source_uri).split("\n").reject do |l|
15       l.strip.start_with?('#') or l.strip == ""
16     end
17   end.flatten
18 end
19
20 def parse_debian_sources(data)
21   data.split("\n").push('').inject({}) do |pkgs, l|
22     l.rstrip!
23     pkg = pkgs[:_tmp] || {}
24     if pkg[:_tmp]
25       if l[0..0] == ' '
26         pkg[:_tmp][:data] << l.strip
27       else
28         pkg[pkg[:_tmp][:hdr]] = pkg[:_tmp][:data].compact
29         pkg.delete :_tmp
30       end
31     end
32     unless pkg[:_tmp]
33       if l == '':
34         pkgs[pkg['Package'][0]] = pkg
35         pkg = nil
36       else
37         hdr, data = l.strip.split(':', 2)
38         data.strip! unless data.nil?
39         pkg[:_tmp] = {:hdr => hdr, :data => [data]}
40       end
41     end
42     if pkg.nil?
43       pkgs.delete :_tmp
44     else
45       pkgs[:_tmp] = pkg
46     end
47     pkgs
48   end
49 end
50
51 def build_package_list(repos, used, sources)
52   data = {}
53   (repos.keys + sources.keys).each do |pkg|
54     p = {
55       :head_is_tagged => false,
56       :used => {},
57       :name => pkg,
58       :git_version => nil,
59       :has_tags => false,
60       :repo_version => nil,
61     }
62
63     begin
64       g = Git.bare(working_dir = repos[pkg])
65       tree = g.ls_tree('HEAD')["tree"]
66       if pkg == 'grml-kernel'
67         tree = g.ls_tree(g.ls_tree("HEAD")["tree"]["linux-3"][:sha])["tree"]
68       end
69       current_head = g.gcommit('HEAD')
70       raise "no head" if not current_head.parent
71       raise "no debian dir in git" if not tree.keys.include?("debian")
72       p.merge!({
73         :git_browser => "http://git.grml.org/?p=%s.git;a=summary" % pkg,
74         :git_anon => "git://git.grml.org/%s.git" % pkg,
75       })
76     rescue Exception
77     end
78
79     used.each do |dist,l|
80       p[:used][dist] = l.include?(pkg)
81     end
82     if sources[pkg]
83       p.merge!({
84         :repo_url => "http://deb.grml.org/pool/main/%s/%s/" % [sources[pkg]['Package'][0][0..0], sources[pkg]['Package'][0]],
85         :repo_version => sources[pkg]['Version'][0],
86         :source_name => sources[pkg]['Package'][0],
87       })
88     end
89
90     if p[:git_anon]
91       for tag in g.tags.reverse
92         p[:has_tags] = true
93         t = g.gcommit(tag.name)
94         next if not t.parent
95         #$stderr.puts "#{pkg}: Checking tag #{tag.name}: tag parent: #{t.parent.sha} HEAD: #{current_head.parent.sha}"
96         if t.parent.sha === current_head.parent.sha
97           p[:head_is_tagged] = true
98           p[:git_version] = tag.name
99           break
100         end
101       end
102     end
103
104     data[pkg] = p
105   end
106   data
107 end
108
109 def fetch_file(uri)
110   response = Net::HTTP.get_response(URI.parse(uri))
111   body = response.body
112   if uri.match(/\.gz$/)
113     body = Zlib::GzipReader.new(StringIO.new(body.to_s)).read
114   end
115   body
116 end
117
118 def update_git_repos(git_repos)
119   git_repos.each do |name, path|
120     out = %x{cd #{path} && git remote update --prune 2>&1}
121     puts "#{name}: " + out if DEBUG
122   end
123 end
124
125 template = ERB.new <<-EOF
126 <!doctype html>
127 <head>
128   <meta charset="utf-8">
129   <title>Grml.org Package Index</title>
130   <link rel="stylesheet" href="style.css">
131 </head>
132 <body>
133   <header>
134   <h1>All Grml packages</h1>
135   </header>
136   <div id="main">
137   <table>
138   <tr>
139   <th>Package</th><th>Git</th><th>Download</th><th>Git Version</th><th>grml-testing</th><th>In FULL?</th>
140   </tr>
141   <% packages.keys.sort.each do |pn|
142   p = packages[pn]
143   %>
144   <tr>
145   <td><%= p[:name] %></td>
146   <td class="git"><% if p[:git_browser] %><a href="<%= p[:git_browser] %>">Git</a><% end %></td>
147   <td class="download"><% if p[:has_tags] %><a href="<%= p[:repo_url] %>">Download</a><% end %></td>
148   <% if p[:head_is_tagged] %>
149   <td class="ok">Version <%= p[:git_version] %></td>
150   <% else %>
151   <td class="error <% if p[:used][:full] %>important<% end %>">Untagged changes</td>
152   <% end %>
153   <td><%= p[:repo_version] || "" %></td>
154   <td class="installed"><%= p[:used][:full] ? "Yes" : "No" %></td>
155   </tr>
156   <% end %>
157   </table>
158   </div>
159   <div>
160   Other packages: <%= other_packages.join(" ") %>
161   </div>
162   <footer>
163   <p>Last update: <%= Time.now.to_s %></p>
164   </footer>
165 </body>
166 </html>
167 EOF
168
169 used_packages = {
170   :full => build_used_package_list([
171                                     'http://git.grml.org/?p=grml-live.git;a=blob_plain;f=etc/grml/fai/config/package_config/GRMLBASE',
172                                     'http://git.grml.org/?p=grml-live.git;a=blob_plain;f=etc/grml/fai/config/package_config/GRML_FULL',
173                                    ]),
174 }
175 sources = {}
176 parse_debian_sources(fetch_file('http://deb.grml.org/dists/grml-testing/main/source/Sources.gz')).each do |k,v|
177   if v['Vcs-Git'] and v['Vcs-Git'][0]
178     m = v['Vcs-Git'][0].match 'git.grml.org\/(.*).git$'
179     k = m[1] if m
180   end
181   sources[k] = v
182 end
183
184 git_repos = Hash[*(Dir.glob('git/*.git').map do |p| [File.basename(p, '.git'), p] end.flatten)]
185
186 update_git_repos git_repos
187
188 packages = build_package_list(git_repos, used_packages, sources)
189 other_packages = (sources.keys - git_repos.keys)
190
191 File.open('index.html.new','w') do |f|
192   f.write template.result(binding)
193 end
194 File.open('packages.yaml.new','w') do |f|
195   f.write packages.to_yaml
196 end
197
198 FileUtils.mv 'index.html.new', 'index.html'
199 FileUtils.mv 'packages.yaml.new', 'packages.yaml'
200