Skip to content
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions errors/v2.yml
Original file line number Diff line number Diff line change
Expand Up @@ -613,6 +613,11 @@
http_code: 502
message: "Deletion of app %s failed because one or more associated resources could not be deleted.\n\n%s"

150010:
name: ResourceChecksumMismatch
http_code: 500
message: "One or more cached resources did not match the given checksum. They have been deleted; please retry package upload."

160001:
name: AppBitsUploadInvalid
http_code: 400
Expand Down
82 changes: 79 additions & 3 deletions lib/cloud_controller/packager/local_bits_packer.rb
Original file line number Diff line number Diff line change
@@ -1,15 +1,13 @@
require 'cloud_controller/blobstore/fingerprints_collection'
require 'cloud_controller/app_packager'
require 'cloud_controller/packager/shared_bits_packer'
require 'shellwords'

module CloudController
module Packager
class ConflictError < StandardError
end

class LocalBitsPacker
include Packager::SharedBitsPacker

def send_package_to_blobstore(blobstore_key, uploaded_package_zip, cached_files_fingerprints)
tmp_dir = VCAP::CloudController::Config.config.get(:directories, :tmpdir)
local_bits_packer_path = File.join(tmp_dir, "local_bits_packer-#{blobstore_key}")
Expand Down Expand Up @@ -39,6 +37,84 @@ def send_package_to_blobstore(blobstore_key, uploaded_package_zip, cached_files_
def package_blobstore
CloudController::DependencyLocator.instance.package_blobstore
end

def package_zip_exists?(package_zip)
package_zip && File.exist?(package_zip)
end

def validate_size!(app_packager)
return unless max_package_size

return unless app_packager.size > max_package_size

raise CloudController::Errors::ApiError.new_from_details('AppPackageInvalid', "Package may not be larger than #{max_package_size} bytes")
end

def copy_uploaded_package(uploaded_package_zip, app_packager)
FileUtils.chmod('u+w', uploaded_package_zip)
FileUtils.cp(uploaded_package_zip, app_packager.path)
end

def populate_resource_cache(app_packager, app_contents_path)
FileUtils.mkdir(app_contents_path)
app_packager.unzip(app_contents_path)

remove_symlinks(app_contents_path)

global_app_bits_cache.cp_r_to_blobstore(app_contents_path)
end

def remove_symlinks(app_contents_path)
Find.find(app_contents_path) do |path|
File.delete(path) if File.symlink?(path)
end
end

def append_matched_resources(app_packager, cached_files_fingerprints, root_path)
matched_resources = CloudController::Blobstore::FingerprintsCollection.new(cached_files_fingerprints, root_path)
cached_resources_dir = File.join(root_path, 'cached_resources_dir')

FileUtils.mkdir(cached_resources_dir)
checksum_mismatch = false
matched_resources.each do |local_destination, file_sha, mode|
file_path = File.join(cached_resources_dir, local_destination)
global_app_bits_cache.download_from_blobstore(file_sha, File.join(cached_resources_dir, local_destination), mode:)

if Digester.new.digest_path(file_path) != file_sha
checksum_mismatch = true
global_app_bits_cache.delete(file_sha)
end
end

raise CloudController::Errors::ApiError.new_from_details('ResourceChecksumMismatch') if checksum_mismatch

app_packager.append_dir_contents(cached_resources_dir)
end

def match_resources_and_validate_package(root_path, uploaded_package_zip, cached_files_fingerprints)
app_packager = AppPackager.new(File.join(root_path, 'copied_app_package.zip'))
app_contents_path = File.join(root_path, 'application_contents')

if package_zip_exists?(uploaded_package_zip)
copy_uploaded_package(uploaded_package_zip, app_packager)
validate_size!(app_packager)
populate_resource_cache(app_packager, app_contents_path) unless VCAP::CloudController::FeatureFlag.disabled?(:resource_matching)
end

append_matched_resources(app_packager, cached_files_fingerprints, root_path)

validate_size!(app_packager)
app_packager.fix_subdir_permissions(root_path, app_contents_path)
app_packager.path
end

def max_package_size
VCAP::CloudController::Config.config.get(:packages, :max_package_size)
end

def global_app_bits_cache
CloudController::DependencyLocator.instance.global_app_bits_cache
end
end
end
end
77 changes: 0 additions & 77 deletions lib/cloud_controller/packager/shared_bits_packer.rb

This file was deleted.

4 changes: 2 additions & 2 deletions spec/api/documentation/app_bits_api_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -35,8 +35,8 @@

let(:fingerprints) do
[
{ fn: 'path/to/content.txt', size: 123, sha1: 'b907173290db6a155949ab4dc9b2d019dea0c901' },
{ fn: 'path/to/code.jar', size: 123, sha1: 'ff84f89760317996b9dd180ab996b079f418396f' }
{ fn: 'path/to/content.txt', size: 123, sha1: 'da39a3ee5e6b4b0d3255bfef95601890afd80709' },
{ fn: 'path/to/code.jar', size: 123, sha1: 'da39a3ee5e6b4b0d3255bfef95601890afd80709' }
]
end

Expand Down
2 changes: 1 addition & 1 deletion spec/request/v2/apps_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -1649,7 +1649,7 @@ def make_actual_lrp(instance_guid:, index:, state:, error:, since:)
TestConfig.config[:directories][:tmpdir] = File.dirname(valid_zip.path)
upload_params = {
application: valid_zip,
resources: [{ fn: 'a/b/c', size: 1, sha1: 'sha' }].to_json
resources: [{ fn: 'a/b/c', size: 1, sha1: 'da39a3ee5e6b4b0d3255bfef95601890afd80709' }].to_json
}
put "/v2/apps/#{process.guid}/bits", upload_params, headers_for(user)
end
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -117,7 +117,7 @@ def make_request
end

context 'with at least one resource and no application' do
let(:req_body) { { resources: Oj.dump([{ 'fn' => 'lol', 'sha1' => 'abc', 'size' => 2048 }]) } }
let(:req_body) { { resources: Oj.dump([{ 'fn' => 'lol', 'sha1' => 'da39a3ee5e6b4b0d3255bfef95601890afd80709', 'size' => 2048 }]) } }

before do
# rack_test overrides 'CONTENT_TYPE' header with 'boundary' which causes errors when the request does not contain an application
Expand All @@ -132,7 +132,7 @@ def make_request
end

context 'with at least one resource and an application' do
let(:req_body) { { resources: Oj.dump([{ 'fn' => 'lol', 'sha1' => 'abc', 'size' => 2048 }]), application: valid_zip } }
let(:req_body) { { resources: Oj.dump([{ 'fn' => 'lol', 'sha1' => 'da39a3ee5e6b4b0d3255bfef95601890afd80709', 'size' => 2048 }]), application: valid_zip } }

it 'succeeds to upload' do
make_request
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ module CloudController::Packager
)
end

let(:fingerprints) do
let(:corrupted_fingerprints) do
path = File.join(local_tmp_dir, 'content')
sha = 'some_fake_sha'
File.write(path, 'content')
Expand All @@ -35,6 +35,15 @@ module CloudController::Packager
[{ 'fn' => 'path/to/content.txt', 'size' => 123, 'sha1' => sha }]
end

let(:fingerprints) do
path = File.join(local_tmp_dir, 'content')
File.write(path, 'content')
sha = Digester.new.digest_path(path)
global_app_bits_cache.cp_to_blobstore(path, sha)

[{ 'fn' => 'path/to/content.txt', 'size' => 123, 'sha1' => sha }]
end

before do
TestConfig.override(directories: { tmpdir: local_tmp_dir })

Expand Down Expand Up @@ -136,6 +145,18 @@ module CloudController::Packager
expect(package_blobstore.exists?(blobstore_key)).to be true
end
end

context 'and there are corrupted cached files' do
let(:cached_files_fingerprints) { corrupted_fingerprints }

it 'deletes the offending files from the blobstore and errors with a ChecksumMismatch error' do
expect(global_app_bits_cache).to receive(:delete).with('some_fake_sha')
expect do
packer.send_package_to_blobstore(blobstore_key, uploaded_files_path, cached_files_fingerprints)
end.
to raise_error(CloudController::Errors::ApiError, /One or more cached resources/)
end
end
end

context 'when the zip file is invalid' do
Expand Down