Class: Jets::Builders::RubyPackager

Inherits:
Object
  • Object
show all
Includes:
Util
Defined in:
lib/jets/builders/ruby_packager.rb

Direct Known Subclasses

RackPackager

Constant Summary collapse

GEM_REGEXP =
/-(arm|x)\d+.*-(darwin|linux)/

Instance Attribute Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(relative_app_root) ⇒ RubyPackager

Returns a new instance of RubyPackager.



10
11
12
# File 'lib/jets/builders/ruby_packager.rb', line 10

def initialize(relative_app_root)
  @full_app_root = "#{build_area}/#{relative_app_root}"
end

Instance Attribute Details

#full_app_rootObject (readonly)

Returns the value of attribute full_app_root.



9
10
11
# File 'lib/jets/builders/ruby_packager.rb', line 9

def full_app_root
  @full_app_root
end

Instance Method Details

#bundle_checkObject

Example ‘bundle check` error:

The following gems are missing
* date (3.3.3)
* timeout (0.3.2)
Install missing gems with `bundle install`

Example success:

The Gemfile's dependencies are satisfied


94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
# File 'lib/jets/builders/ruby_packager.rb', line 94

def bundle_check
  out = ''
  Bundler.with_unbundled_env do
    out = `cd #{cache_area} && bundle check 2>&1`
  end
  if out.include?("missing")
    puts "Failed: bundle check".color(:red)
    puts <<~EOL
      This means something went wrong with the bundle install.
      Jets will prevent the deployment to AWS Lambda.
      It's better to error now instead of finding out on AWS Lambda.
      The bundle install can fail for different system-specific reasons.
      It could be an outdated or incompatible version of RubyGems and Ruby.

      Related: https://community.boltops.com/t/could-not-find-timeout-0-3-1-in-any-of-the-sources/996

    EOL
    exit 1
  end
end

#bundle_installObject

Installs gems on the current target system: both compiled and non-compiled. If user is on a macosx machine, macosx gems will be installed. If user is on a linux machine, linux gems will be installed.

Copies Gemfile* to /tmp/jets/demo/cache folder and installs gems with bundle install from there.

We take the time to copy Gemfile and bundle into a separate directory because it gets left around to act as a ‘cache’. So, when the builds the project gets built again not all the gems from get installed from the beginning.



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
# File 'lib/jets/builders/ruby_packager.rb', line 46

def bundle_install
  full_project_path = @full_app_root
  headline "Bundling: running bundle install in cache area: #{cache_area}."

  copy_gemfiles(full_project_path)
  copy_bundled_gems(full_project_path)
  run_prebundle_copy

  # Uncomment out to always remove the cache/vendor/gems to debug
  # FileUtils.rm_rf("#{cache_area}/vendor/gems")

  # Remove .bundle folder so .bundle/config doesnt affect how Jets packages gems.
  # Not using BUNDLE_IGNORE_CONFIG=1 to allow home ~/.bundle/config to affect bundling though.
  # This is useful if you have private gems sources that require authentication. Example:
  #
  #    bundle config gems.myprivatesource.com user:pass
  #

  create_bundle_config
  require "bundler" # dynamically require bundler so user can use any bundler
  Bundler.with_unbundled_env do
    sh(
      "cd #{cache_area} && " \
      "env bundle install"
    )
  end
  create_bundle_config(frozen: true)

  rewrite_gemfile_lock("#{cache_area}/Gemfile.lock")

  # Copy the Gemfile.lock back to the project in case it was updated.
  # For example we add the jets-rails to the Gemfile.
  copy_back_gemfile_lock

  puts 'Bundle install completed'
end

#clean_old_submodulesObject

When using submodules, bundler leaves old submodules behind. Over time this inflates the size of the the cache gems. So we’ll clean it up.



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
# File 'lib/jets/builders/ruby_packager.rb', line 136

def clean_old_submodules
  # https://stackoverflow.com/questions/38800129/parsing-a-gemfile-lock-with-bundler
  lockfile = "#{cache_area}/Gemfile.lock"
  return unless File.exist?(lockfile)

  return if Bundler.bundler_major_version <= 1 # LockfileParser only works for Bundler version 2+

  parser = Bundler::LockfileParser.new(Bundler.read_file(lockfile))
  specs = parser.specs

  # specs = Bundler.load.specs
  # IE: spec.source.to_s: "https://github.com/tongueroo/webpacker.git (at jets@a8c4661)"
  submoduled_specs = specs.select do |spec|
    spec.source.to_s =~ /@\w+\)/
  end

  # find git shas to keep
  # IE: ["a8c4661", "abc4661"]
  git_shas = submoduled_specs.map do |spec|
    md = spec.source.to_s.match(/@(\w+)\)/)
    md[1] # git_sha
  end

  # IE: /tmp/jets/demo/cache/vendor/gems/ruby/2.5.0/bundler/gems/webpacker-a8c46614c675
  Dir.glob("#{cache_area}/vendor/gems/ruby/2.5.0/bundler/gems/*").each do |path|
    sha = path.split('-').last[0..6] # only first 7 chars of the git sha
    unless git_shas.include?(sha)
      FileUtils.rm_rf(path) # REMOVE old submodule directory
    end
  end
end

#copy_back_gemfile_lockObject



115
116
117
118
119
# File 'lib/jets/builders/ruby_packager.rb', line 115

def copy_back_gemfile_lock
  src = "#{cache_area}/Gemfile.lock"
  dest = "#{@full_app_root}/Gemfile.lock"
  FileUtils.cp(src, dest)
end

#copy_bundle_configObject



247
248
249
250
251
252
253
254
255
256
# File 'lib/jets/builders/ruby_packager.rb', line 247

def copy_bundle_config
  # Override project's .bundle/config and ensure that .bundle/config matches
  # at these 2 spots:
  #   app_root/.bundle/config
  #   vendor/gems/.bundle/config
  cache_bundle_config = "#{cache_area}/.bundle/config"
  app_bundle_config = "#{@full_app_root}/.bundle/config"
  FileUtils.mkdir_p(File.dirname(app_bundle_config))
  FileUtils.cp(cache_bundle_config, app_bundle_config)
end

#copy_bundled_gems(full_project_path) ⇒ Object



180
181
182
183
184
# File 'lib/jets/builders/ruby_packager.rb', line 180

def copy_bundled_gems(full_project_path)
  src = "#{full_project_path}/bundled_gems"
  return unless File.exist?(src)
  Jets::Util.cp_r(src, "#{cache_area}/bundled_gems")
end

#copy_cache_gemsObject



275
276
277
278
279
280
281
282
283
284
285
286
# File 'lib/jets/builders/ruby_packager.rb', line 275

def copy_cache_gems
  vendor_gems = "#{@full_app_root}/vendor/gems"
  if File.exist?(vendor_gems)
    puts "Removing current vendor_gems from project"
    FileUtils.rm_rf(vendor_gems)
  end
  # Leave #{Jets.build_root}/vendor_gems behind to act as cache
  if File.exist?("#{cache_area}/vendor/gems")
    FileUtils.mkdir_p(File.dirname(vendor_gems))
    Jets::Util.cp_r("#{cache_area}/vendor/gems", vendor_gems)
  end
end

#copy_gemfiles(full_project_path) ⇒ Object



169
170
171
172
173
174
175
176
177
178
# File 'lib/jets/builders/ruby_packager.rb', line 169

def copy_gemfiles(full_project_path)
  FileUtils.mkdir_p(cache_area)
  FileUtils.cp("#{full_project_path}/Gemfile", "#{cache_area}/Gemfile")

  gemfile_lock = "#{full_project_path}/Gemfile.lock"
  dest = "#{cache_area}/Gemfile.lock"
  return unless File.exist?(gemfile_lock)

  FileUtils.cp(gemfile_lock, dest)
end

#create_bundle_config(frozen: false) ⇒ Object

On circleci the “#Jets.build_root/.bundle/config” doesnt exist this only happens with ssh debugging, not when the ci.sh script gets ran. But on macosx it exists. Dont know why this is the case.



262
263
264
265
266
267
268
269
270
271
272
273
# File 'lib/jets/builders/ruby_packager.rb', line 262

def create_bundle_config(frozen: false)
  FileUtils.rm_rf("#{cache_area}/.bundle")
  frozen_line = %Q|BUNDLE_FROZEN: "true"\n| if frozen
  text =<<-EOL
---
#{frozen_line}BUNDLE_PATH: "vendor/gems"
BUNDLE_WITHOUT: "development:test"
EOL
  bundle_config = "#{cache_area}/.bundle/config"
  FileUtils.mkdir_p(File.dirname(bundle_config))
  IO.write(bundle_config, text)
end

#finishObject

build gems in vendor/gems/ruby/2.5.0 (done in install phase)



25
26
27
28
# File 'lib/jets/builders/ruby_packager.rb', line 25

def finish
  return unless gemfile_exist?
  tidy
end

#gemfile_exist?Boolean

Returns:

  • (Boolean)


30
31
32
33
# File 'lib/jets/builders/ruby_packager.rb', line 30

def gemfile_exist?
  gemfile_path = "#{@full_app_root}/Gemfile"
  File.exist?(gemfile_path)
end

#installObject



14
15
16
17
18
19
20
21
22
# File 'lib/jets/builders/ruby_packager.rb', line 14

def install
  return unless gemfile_exist?

  clean_old_submodules
  bundle_install
  bundle_check
  copy_bundle_config
  copy_cache_gems
end

#rewrite_gemfile_lock(gemfile_lock) ⇒ Object

Remove the BUNDLED WITH line since we don’t control the bundler gem version on AWS Lambda And this can cause issues with require ‘bundler/setup’



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
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
# File 'lib/jets/builders/ruby_packager.rb', line 198

def rewrite_gemfile_lock(gemfile_lock)
  lines = IO.readlines(gemfile_lock)

  # Remove BUNDLED WITH
  # amount is the number of lines to remove
  new_lines, capture, count, amount = [], true, 0, 2
  lines.each do |l|
    capture = false if l.include?('BUNDLED WITH')
    if capture
      new_lines << l
    end
    if capture == false
      count += 1
      capture = count > amount # renable capture
    end
  end

  # Replace things like nokogiri (1.11.1-x86_64-darwin) => nokogiri (1.11.1)
  lines, new_lines = new_lines, []
  lines.each do |l|
    l.sub!(GEM_REGEXP, '') if l =~ GEM_REGEXP
    new_lines << l
  end

  # Make sure platform is ruby
  lines, new_lines, in_platforms_section, platforms_rewritten = new_lines, [], false, false
  lines.each do |l|
    if in_platforms_section && platforms_rewritten # once PLATFORMS has been found, skip all lines until the next section
      if l.present?
        next
      else
        in_platforms_section = false
      end
    end

    if in_platforms_section && !platforms_rewritten # specify ruby as the only platform
      new_lines << "  ruby\n"
      platforms_rewritten = true
      next
    end

    in_platforms_section = l.include?('PLATFORMS')
    new_lines << l
  end

  content = new_lines.join('')
  IO.write(gemfile_lock, content)
end

#run_prebundle_copyObject



186
187
188
189
190
191
192
193
194
# File 'lib/jets/builders/ruby_packager.rb', line 186

def run_prebundle_copy
  paths = Jets.config.build.prebundle_copy
  paths = Array(paths)
  paths.each do |path|
    src = "#{@full_app_root}/#{path}"
    return unless File.exist?(src)
    Jets::Util.cp_r(src, "#{cache_area}/#{path}")
  end
end

#tidyObject

Clean up extra unneeded files to reduce package size Because we’re removing files (something dangerous) use full paths.



123
124
125
126
127
128
# File 'lib/jets/builders/ruby_packager.rb', line 123

def tidy
  puts "Tidying project: removing ignored files to reduce package size."
  tidy_project(@full_app_root)
  # The rack sub project has it's own gitignore.
  tidy_project(@full_app_root+"/rack")
end

#tidy_project(path) ⇒ Object



130
131
132
# File 'lib/jets/builders/ruby_packager.rb', line 130

def tidy_project(path)
  Tidy.new(path).cleanup!
end