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
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
|
require 'formula'
require 'bottles'
require 'tab'
require 'keg'
require 'cmd/versions'
require 'utils/inreplace'
require 'erb'
require 'open3'
require 'extend/pathname'
class BottleMerger < Formula
# This provides a URL and Version which are the only needed properties of
# a Formula. This object is used to access the Formula bottle DSL to merge
# multiple outputs of `brew bottle`.
url '1'
def self.reset_bottle; @bottle = Bottle.new; end
end
BOTTLE_ERB = <<-EOS
bottle do
<% if prefix.to_s != '/usr/local' %>
prefix '<%= prefix %>'
<% end %>
<% if cellar.is_a? Symbol %>
cellar :<%= cellar %>
<% elsif cellar.to_s != '/usr/local/Cellar' %>
cellar '<%= cellar %>'
<% end %>
<% if revision > 0 %>
revision <%= revision %>
<% end %>
<% checksums.each do |checksum_type, checksum_values| %>
<% checksum_values.each do |checksum_value| %>
<% checksum, osx = checksum_value.shift %>
<%= checksum_type %> '<%= checksum %>' => :<%= osx %>
<% end %>
<% end %>
end
EOS
module Homebrew extend self
class << self
include Utils::Inreplace
end
def keg_contains string, keg
if not ARGV.homebrew_developer?
return quiet_system 'fgrep', '--recursive', '--quiet', '--max-count=1', string, keg
end
# Find all files that still reference the keg via a string search
keg_ref_files = `/usr/bin/fgrep --files-with-matches --recursive "#{string}" "#{keg}" 2>/dev/null`
keg_ref_files = (keg_ref_files.map{ |file| Pathname.new(file.strip) }).reject(&:symlink?)
# If there are no files with that string found, return immediately
return false if keg_ref_files.empty?
# Start printing out each file and any extra information we can find
opoo "String '#{string}' still exists in these files:"
keg_ref_files.each do |file|
puts "#{Tty.red}#{file}#{Tty.reset}"
# If we can't use otool on this file, just skip to the next file
next if not file.mach_o_executable? and not file.mach_o_bundle? and not file.dylib? and not file.extname == '.a'
# Get all libraries this file links to, then display only links to libraries that contain string in the path
linked_libraries = `otool -L "#{file}"`.split("\n").drop(1)
linked_libraries.map!{ |lib| lib.strip.split()[0] }
linked_libraries = linked_libraries.select{ |lib| lib.include? string }
linked_libraries.each do |lib|
puts " #{Tty.gray}-->#{Tty.reset} links to #{lib}"
end
# Use strings to search through the file for each string
strings = `strings -t x - "#{file}"`.select{ |str| str.include? string }.map{ |s| s.strip }
# Don't bother reporting a string if it was found by otool
strings.reject!{ |str| linked_libraries.include? str.split[1] }
strings.each do |str|
offset, match = str.split
puts " #{Tty.gray}-->#{Tty.reset} match '#{match}' at offset #{Tty.em}0x#{offset}#{Tty.reset}"
end
end
puts
true
end
def bottle_output bottle
erb = ERB.new BOTTLE_ERB
erb.result(bottle.instance_eval { binding }).gsub(/^\s*$\n/, '')
end
def bottle_formula f
unless f.installed?
return ofail "Formula not installed: #{f.name}"
end
unless built_as_bottle? f
return ofail "Formula not installed with '--build-bottle': #{f.name}"
end
master_bottle_filenames = f.bottle_filenames 'origin/master'
bottle_revision = -1
begin
bottle_revision += 1
filename = bottle_filename(f, bottle_revision)
end while not ARGV.include? '--no-revision' \
and master_bottle_filenames.include? filename
if bottle_filename_formula_name(filename).empty?
return ofail "Add a new regex to bottle_version.rb to parse the bottle filename."
end
bottle_path = Pathname.pwd/filename
sha1 = nil
prefix = HOMEBREW_PREFIX.to_s
tmp_prefix = '/tmp'
cellar = HOMEBREW_CELLAR.to_s
tmp_cellar = '/tmp/Cellar'
output = nil
HOMEBREW_CELLAR.cd do
ohai "Bottling #{filename}..."
# Use gzip, faster to compress than bzip2, faster to uncompress than bzip2
# or an uncompressed tarball (and more bandwidth friendly).
safe_system 'tar', 'czf', bottle_path, "#{f.name}/#{f.version}"
sha1 = bottle_path.sha1
relocatable = false
if File.size?(bottle_path) > 1*1024*1024
ohai "Detecting if #{filename} is relocatable..."
end
keg = Keg.new f.prefix
keg.lock do
# Relocate bottle library references before testing for built-in
# references to the Cellar e.g. Qt's QMake annoyingly does this.
keg.relocate_install_names prefix, tmp_prefix, cellar, tmp_cellar, :keg_only => f.keg_only?
if prefix == '/usr/local'
prefix_check = HOMEBREW_PREFIX/'opt'
else
prefix_check = HOMEBREW_PREFIX
end
relocatable = !keg_contains(prefix_check, keg)
relocatable = !keg_contains(HOMEBREW_CELLAR, keg) if relocatable
# And do the same thing in reverse to change the library references
# back to how they were.
keg.relocate_install_names tmp_prefix, prefix, tmp_cellar, cellar
end
bottle = Bottle.new
bottle.prefix HOMEBREW_PREFIX
bottle.cellar relocatable ? :any : HOMEBREW_CELLAR
bottle.revision bottle_revision
bottle.sha1 sha1 => bottle_tag
puts "./#{filename}"
output = bottle_output bottle
puts output
end
if ARGV.include? '--rb'
bottle_base = filename.gsub(bottle_suffix(bottle_revision), '')
File.open "#{bottle_base}.bottle.rb", 'w' do |file|
file.write output
end
end
end
def merge
merge_hash = {}
ARGV.named.each do |argument|
formula_name = bottle_filename_formula_name argument
merge_hash[formula_name] ||= []
bottle_block = IO.read argument
merge_hash[formula_name] << bottle_block
end
merge_hash.keys.each do |formula_name|
BottleMerger.reset_bottle
ohai formula_name
bottle_blocks = merge_hash[formula_name]
bottle_blocks.each do |bottle_block|
BottleMerger.class_eval bottle_block
end
bottle = BottleMerger.new.bottle
next unless bottle
output = bottle_output bottle
puts output
if ARGV.include? '--write'
f = Formula.factory formula_name
formula_relative_path = "Library/Formula/#{f.name}.rb"
if File.exists? formula_relative_path
formula_path = Pathname.new(formula_relative_path)
else
formula_path = HOMEBREW_REPOSITORY+formula_relative_path
end
inreplace formula_path do |s|
if f.bottle
s.gsub!(/ bottle do.+?end\n/m, output)
else
s.gsub!(/( (url|sha1|head|version) '\S*'\n+)+/m, '\0' + output + "\n")
end
end
update_or_add = f.bottle.nil? ? 'add' : 'update'
safe_system 'git', 'commit', formula_path, '-m',
"#{f.name}: #{update_or_add} bottle."
end
end
exit 0
end
def bottle
merge if ARGV.include? '--merge'
ARGV.formulae.each do |f|
bottle_formula f
end
end
end
|