@Override public PutObjectResult putObject(PutObjectRequest request) { if (request.getRange() != null) throw new UnsupportedOperationException(PARTIAL_UPDATE_MSG); // make user metadata available as a request property if (request.getObjectMetadata() == null) request.setObjectMetadata(new S3ObjectMetadata()); Map<String, String> userMeta = request.getObjectMetadata().getUserMetadata(); request.property(RestUtil.PROPERTY_USER_METADATA, userMeta); // activate codec filter request.property(RestUtil.PROPERTY_ENCODE_ENTITY, Boolean.TRUE); // write data PutObjectResult result = super.putObject(request); // encryption filter will modify userMeta with encryption metadata *after* the object is transferred // we must send a separate metadata update or the object will be unreadable // TODO: should this be atomic? how do we handle rollback? CopyObjectRequest metadataUpdate = new CopyObjectRequest(request.getBucketName(), request.getKey(), request.getBucketName(), request.getKey()).withAcl(request.getAcl()) .withObjectMetadata(request.getObjectMetadata()).withIfMatch(result.getETag()); return super.copyObject(metadataUpdate); }
@Override public PutObjectResult putObject(PutObjectRequest request) { if (request.getObjectMetadata() != null) { request.setObjectMetadata(null); } try { Path path = Paths.get(this.baseDir, request.getBucketName(), request.getKey()); Path parent = path.getParent(); assert parent != null; Files.createDirectories(parent); Files.createFile(path); } catch (IOException e) { throw new S3Exception(e.getMessage(), 0, e); } PutObjectResult retVal = new PutObjectResult(); if (request.getAcl() != null) { long size = 0; if (request.getRange() != null) { size = request.getRange().getLast() + 1; } aclMap.putIfAbsent(request.getKey(), new AclSize(request.getAcl(), size)); } return retVal; }