com.obs.services.internal.service.ObsObjectBaseService Maven / Gradle / Ivy
/**
* Copyright 2019 Huawei Technologies Co.,Ltd.
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use
* this file except in compliance with the License. You may obtain a copy of the
* License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed
* under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
* CONDITIONS OF ANY KIND, either express or implied. See the License for the
* specific language governing permissions and limitations under the License.
*/
package com.obs.services.internal.service;
import com.obs.log.ILogger;
import com.obs.log.LoggerBuilder;
import com.obs.services.exception.ObsException;
import com.obs.services.internal.Constants;
import com.obs.services.internal.Constants.CommonHeaders;
import com.obs.services.internal.Constants.ObsRequestParams;
import com.obs.services.internal.ObsConstraint;
import com.obs.services.internal.ProgressManager;
import com.obs.services.internal.ServiceException;
import com.obs.services.internal.SimpleProgressManager;
import com.obs.services.internal.handler.XmlResponsesSaxParser;
import com.obs.services.internal.io.HttpMethodReleaseInputStream;
import com.obs.services.internal.io.ProgressInputStream;
import com.obs.services.internal.trans.NewTransResult;
import com.obs.services.internal.utils.JSONChange;
import com.obs.services.internal.utils.Mimetypes;
import com.obs.services.internal.utils.RestUtils;
import com.obs.services.internal.utils.ServiceUtils;
import com.obs.services.model.AccessControlList;
import com.obs.services.model.AuthTypeEnum;
import com.obs.services.model.CopyObjectRequest;
import com.obs.services.model.CopyObjectResult;
import com.obs.services.model.DeleteObjectRequest;
import com.obs.services.model.DeleteObjectResult;
import com.obs.services.model.DeleteObjectsRequest;
import com.obs.services.model.DeleteObjectsResult;
import com.obs.services.model.GetObjectAclRequest;
import com.obs.services.model.GetObjectMetadataRequest;
import com.obs.services.model.GetObjectRequest;
import com.obs.services.model.HeaderResponse;
import com.obs.services.model.HttpMethodEnum;
import com.obs.services.model.ObjectMetadata;
import com.obs.services.model.ObsObject;
import com.obs.services.model.PutObjectInTwoBucketRequest;
import com.obs.services.model.PutObjectInTwoBucketResult;
import com.obs.services.model.PutObjectRequest;
import com.obs.services.model.PutObjectResult;
import com.obs.services.model.SetObjectAclRequest;
import com.obs.services.model.SetObjectMetadataRequest;
import com.obs.services.model.SpecialParamEnum;
import com.obs.services.model.StorageClassEnum;
import com.obs.services.model.fs.DropFileResult;
import com.obs.services.model.fs.ObsFSAttribute;
import com.obs.services.model.fs.ObsFSFile;
import com.obs.services.model.fs.ReadFileResult;
import com.obs.services.model.select.SelectObjectRequest;
import com.obs.services.model.select.SelectObjectResult;
import okhttp3.Request;
import okhttp3.RequestBody;
import okhttp3.Response;
import okhttp3.ResponseBody;
import java.io.BufferedInputStream;
import java.io.Closeable;
import java.nio.charset.StandardCharsets;
import java.text.ParseException;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import static com.obs.services.internal.Constants.OBS_HEADER_PREFIX;
import static com.obs.services.internal.Constants.V2_HEADER_PREFIX;
public abstract class ObsObjectBaseService extends ObsBucketAdvanceService {
private static final ILogger log = LoggerBuilder.getLogger(ObsObjectBaseService.class);
protected boolean doesObjectExistImpl(GetObjectMetadataRequest request) throws ServiceException {
Map headers = new HashMap<>();
this.transSseCHeaders(request.getSseCHeader(), headers, this.getIHeaders(request.getBucketName()));
this.transRequestPaymentHeaders(request, headers, this.getIHeaders(request.getBucketName()));
Map params = new HashMap<>();
if (request.getVersionId() != null) {
params.put(ObsRequestParams.VERSION_ID, request.getVersionId());
}
boolean doesObjectExist = false;
try (Response response = performRestHead(request.getBucketName(), request.getObjectKey(), params, headers,
request.getUserHeaders(), request.isEncodeHeaders())) {
if (200 == response.code()) {
doesObjectExist = true;
}
} catch (ServiceException ex) {
if (!(404 == ex.getResponseCode())) {
throw ex;
}
}
return doesObjectExist;
}
protected ObsFSFile putObjectImpl(PutObjectRequest request) throws ServiceException {
TransResult result = null;
Response response;
boolean isExtraAclPutRequired;
AccessControlList acl = request.getAcl();
NewTransResult newTransResult;
try {
result = this.transPutObjectRequest(request);
isExtraAclPutRequired = !prepareRESTHeaderAcl(request.getBucketName(), result.getHeaders(), acl);
if (request.getCallback() != null) {
ServiceUtils.assertParameterNotNull(request.getCallback().getCallbackUrl(),
"callbackUrl is null");
ServiceUtils.assertParameterNotNull(request.getCallback().getCallbackBody(),
"callbackBody is null");
result.getHeaders().put((this.getProviderCredentials().getLocalAuthType(request.getBucketName()) != AuthTypeEnum.OBS
? V2_HEADER_PREFIX : OBS_HEADER_PREFIX) + CommonHeaders.CALLBACK,
ServiceUtils.toBase64(JSONChange.objToJson(request.getCallback()).getBytes(StandardCharsets.UTF_8)));
}
// todo prepareRESTHeaderAcl 也会操作头域,下次重构可以将其合并
newTransResult = transObjectRequestWithResult(result, request);
response = performRequest(newTransResult, true, false, false, false);
} finally {
if (result != null && result.getBody() != null && request.isAutoClose()) {
if (result.getBody() instanceof Closeable) {
ServiceUtils.closeStream((Closeable) result.getBody());
}
}
}
ObsFSFile ret = new ObsFSFile(request.getBucketName(), request.getObjectKey(),
response.header(CommonHeaders.ETAG),
response.header(this.getIHeaders(request.getBucketName()).versionIdHeader()),
StorageClassEnum.getValueFromCode(response.header(this.getIHeaders(request.getBucketName())
.storageClassHeader())), this.getObjectUrl(request.getBucketName(), request.getObjectKey(),
request.getIsIgnorePort()));
if(request.getCallback() != null) {
try {
ret.setCallbackResponseBody(Objects.requireNonNull(response.body()).byteStream());
} catch (Exception e) {
throw new ServiceException(e);
}
}
setHeadersAndStatus(ret, response);
if (isExtraAclPutRequired && acl != null) {
try {
putAclImpl(request.getBucketName(), request.getObjectKey(), acl, null, request.isRequesterPays());
} catch (Exception e) {
if (log.isWarnEnabled()) {
log.warn("Try to set object acl error", e);
}
}
}
return ret;
}
protected PutObjectInTwoBucketResult putObjectInTwoBucketImpl(PutObjectInTwoBucketRequest request) throws ObsException {
ExecutorService executorService = Executors.newFixedThreadPool(2);
PutObjectInTwoBucketResult putObjectInTwoBucketResult;
try {
Future futureMainBucket = executorService.submit(() -> {
PutObjectInTwoBucketResult result = new PutObjectInTwoBucketResult();
try {
PutObjectResult putObjectResult = this.putObjectImpl(request.getMainBucketRequest());
result.setMainBucketResult(putObjectResult);
} catch (ServiceException e) {
result.setMainBucketException(ServiceUtils.changeFromServiceException(e));
} catch (Exception e) {
result.setMainBucketException(ServiceUtils.changeFromException(e));
}
return result;
});
Future futureBackupBucket = executorService.submit(() -> {
PutObjectInTwoBucketResult result = new PutObjectInTwoBucketResult();
try {
PutObjectResult putObjectResult = this.putObjectImpl(request.getBackupBucketRequest());
result.setBackupBucketResult(putObjectResult);
} catch (ServiceException e) {
result.setBackupBucketException(ServiceUtils.changeFromServiceException(e));
} catch (Exception e) {
result.setBackupBucketException(ServiceUtils.changeFromException(e));
}
return result;
});
executorService.shutdown();
executorService.awaitTermination(Long.MAX_VALUE, TimeUnit.MILLISECONDS);
putObjectInTwoBucketResult = futureMainBucket.get();
PutObjectInTwoBucketResult backupBucketResult = futureBackupBucket.get();
putObjectInTwoBucketResult.setBackupBucketResult(backupBucketResult.getBackupBucketResult());
putObjectInTwoBucketResult.setBackupBucketException(backupBucketResult.getBackupBucketException());
} catch (ServiceException e) {
throw ServiceUtils.changeFromServiceException(e);
} catch (Exception e) {
throw ServiceUtils.changeFromException(e);
}
return putObjectInTwoBucketResult;
}
protected ObsObject getObjectImpl(GetObjectRequest request) throws ServiceException {
return (ObsObject) this.getObjectImpl((GetObjectMetadataRequest) request);
}
protected Object getObjectImpl(GetObjectMetadataRequest request) throws ServiceException {
Response response;
GetObjectRequest getRequest = null;
if (!(request instanceof GetObjectRequest)) {
Map headers = new HashMap<>();
this.transSseCHeaders(request.getSseCHeader(), headers, this.getIHeaders(request.getBucketName()));
this.transRequestPaymentHeaders(request, headers, this.getIHeaders(request.getBucketName()));
Map params = new HashMap<>();
if (request.getVersionId() != null) {
params.put(ObsRequestParams.VERSION_ID, request.getVersionId());
}
response = performRestHead(request.getBucketName(), request.getObjectKey(),
params, headers, request.getUserHeaders(), request.isEncodeHeaders());
} else {
getRequest = (GetObjectRequest) request;
TransResult result = this.transGetObjectRequest(getRequest);
if (getRequest.getRequestParameters() != null) {
result.getParams().putAll(getRequest.getRequestParameters());
}
response = performRestGet(request.getBucketName(), request.getObjectKey(), result.getParams(),
result.getHeaders(), request.getUserHeaders(), false, request.isEncodeHeaders());
}
ObsFSAttribute objMetadata = this.getObsFSAttributeFromResponse(request.getBucketName(),
response, request.isEncodeHeaders());
if (!(request instanceof GetObjectRequest)) {
response.close();
return objMetadata;
}
ReadFileResult obsObject = new ReadFileResult();
obsObject.setObjectKey(request.getObjectKey());
obsObject.setBucketName(request.getBucketName());
obsObject.setMetadata(objMetadata);
// pmd error message: CloseResource - Ensure that resources like this
// InputStream object are closed after use
// 该接口是下载对象,需要将流返回给客户(调用方),我们不能关闭这个流
obsObject.setObjectContent(response.body().byteStream()); // NOPMD
if (getRequest.getProgressListener() != null) {
ProgressManager progressManager = new SimpleProgressManager(objMetadata.getContentLength(), 0,
getRequest.getProgressListener(),
getRequest.getProgressInterval() > 0
? getRequest.getProgressInterval() : ObsConstraint.DEFAULT_PROGRESS_INTERVAL);
obsObject.setObjectContent(new ProgressInputStream(obsObject.getObjectContent(), progressManager));
}
int readBufferSize = obsProperties.getIntProperty(ObsConstraint.READ_BUFFER_SIZE,
ObsConstraint.DEFAULT_READ_BUFFER_STREAM);
if (readBufferSize > 0) {
obsObject.setObjectContent(new BufferedInputStream(obsObject.getObjectContent(), readBufferSize));
}
return obsObject;
}
protected SelectObjectResult selectObjectContentImpl(SelectObjectRequest selectRequest)
throws ServiceException {
Map httpHeaders = new HashMap<>();
httpHeaders.put(CommonHeaders.CONTENT_TYPE, Mimetypes.MIMETYPE_XML);
final String bucket = selectRequest.getBucketName();
Map requestParams = new HashMap<>();
requestParams.put("select", null);
requestParams.put("select-type", "2");
String xml = selectRequest.convertToXml();
if (log.isDebugEnabled()) {
log.debug(
String.format(
"selectObjectContentImpl: bucket: %s; object: %s; request XML: %n%s%n",
bucket,
selectRequest.getKey(),
xml));
}
NewTransResult transResult = new NewTransResult();
transResult.setHttpMethod(HttpMethodEnum.POST);
transResult.setBucketName(bucket);
transResult.setObjectKey(selectRequest.getKey());
transResult.setParams(requestParams);
transResult.setBody(createRequestBody(Mimetypes.MIMETYPE_XML, xml));
Request.Builder builder = setupConnection(transResult, false, false);
renameMetadataKeys(bucket, builder, httpHeaders);
Response response = performRequest(
builder.build(), requestParams, bucket, true, false);
ResponseBody responseBody = response.body();
return new SelectObjectResult(responseBody.byteStream());
}
protected DeleteObjectsResult deleteObjectsImpl(DeleteObjectsRequest request) throws ServiceException {
Map requestParams = new HashMap<>();
requestParams.put(SpecialParamEnum.DELETE.getOriginalStringCode(), "");
String xml = this.getIConvertor(request.getBucketName()).transKeyAndVersion(request.getKeyAndVersions(),
request.isQuiet(), request.getEncodingType());
Map headers = new HashMap<>();
headers.put(CommonHeaders.CONTENT_LENGTH, String.valueOf(xml.length()));
headers.put(CommonHeaders.CONTENT_MD5, ServiceUtils.computeMD5(xml));
headers.put(CommonHeaders.CONTENT_TYPE, Mimetypes.MIMETYPE_XML);
transRequestPaymentHeaders(request, headers, this.getIHeaders(request.getBucketName()));
NewTransResult transResult = transRequest(request);
transResult.setParams(requestParams);
transResult.setHeaders(headers);
transResult.setBody(createRequestBody(Mimetypes.MIMETYPE_XML, xml));
Response response = performRequest(transResult, true, false, false, false);
this.verifyResponseContentType(response);
DeleteObjectsResult ret = getXmlResponseSaxParser().parse(new HttpMethodReleaseInputStream(response),
XmlResponsesSaxParser.DeleteObjectsHandler.class, true).getMultipleDeleteResult();
setHeadersAndStatus(ret, response);
return ret;
}
protected DeleteObjectResult deleteObjectImpl(DeleteObjectRequest request) throws ServiceException {
Map requestParameters = new HashMap<>();
if (request.getVersionId() != null) {
requestParameters.put(ObsRequestParams.VERSION_ID, request.getVersionId());
}
Response response = performRestDelete(request.getBucketName(), request.getObjectKey(), requestParameters,
transRequestPaymentHeaders(request, null, this.getIHeaders(request.getBucketName())),
request.getUserHeaders());
DropFileResult result = new DropFileResult(Boolean.valueOf(response.header(this.getIHeaders(
request.getBucketName()).deleteMarkerHeader())), request.getObjectKey(),
response.header(this.getIHeaders(request.getBucketName()).versionIdHeader()));
setHeadersAndStatus(result, response);
return result;
}
protected CopyObjectResult copyObjectImpl(CopyObjectRequest request) throws ServiceException {
TransResult result = this.transCopyObjectRequest(request);
AccessControlList acl = request.getAcl();
// todo prepareRESTHeaderAcl 也会操作头域,下次重构可以将其合并
boolean isExtraAclPutRequired = !prepareRESTHeaderAcl(request.getBucketName(), result.getHeaders(), acl);
NewTransResult newTransResult = transObjectRequestWithResult(result, request);
Response response = performRequest(newTransResult, true, false, false, false);
this.verifyResponseContentType(response);
XmlResponsesSaxParser.CopyObjectResultHandler handler = getXmlResponseSaxParser().parse(
new HttpMethodReleaseInputStream(response), XmlResponsesSaxParser.CopyObjectResultHandler.class, false);
CopyObjectResult copyRet = new CopyObjectResult(handler.getETag(), handler.getLastModified(),
response.header(this.getIHeaders(request.getBucketName()).versionIdHeader()),
response.header(this.getIHeaders(request.getBucketName()).copySourceVersionIdHeader()),
StorageClassEnum.getValueFromCode(response.header(
this.getIHeaders(request.getBucketName()).storageClassHeader())), handler.getCRC64(),
handler.getCRC32());
setHeadersAndStatus(copyRet, response);
if (isExtraAclPutRequired && acl != null) {
if (log.isDebugEnabled()) {
log.debug("Creating object with a non-canned ACL using REST, so an extra ACL Put is required");
}
try {
putAclImpl(request.getDestinationBucketName(), request.getDestinationObjectKey(), acl, null,
request.isRequesterPays());
} catch (Exception e) {
if (log.isWarnEnabled()) {
log.warn("Try to set object acl error", e);
}
}
}
return copyRet;
}
protected ObjectMetadata setObjectMetadataImpl(SetObjectMetadataRequest request) {
TransResult result = this.transSetObjectMetadataRequest(request);
NewTransResult newTransResult = transObjectRequestWithResult(result, request);
Response response = performRequest(newTransResult);
return this.getObsFSAttributeFromResponse(request.getBucketName(), response, request.isEncodeHeaders());
}
protected ObsFSAttribute getObjectMetadataImpl(GetObjectMetadataRequest request) throws ServiceException {
return (ObsFSAttribute) this.getObjectImpl(request);
}
protected HeaderResponse setObjectAclImpl(SetObjectAclRequest request) throws ServiceException {
Map requestParams = new HashMap<>();
requestParams.put(SpecialParamEnum.ACL.getOriginalStringCode(), "");
if (request.getVersionId() != null) {
requestParams.put(ObsRequestParams.VERSION_ID, request.getVersionId());
}
RequestBody entity = null;
if (ServiceUtils.isValid(request.getCannedACL())) {
request.setAcl(this.getIConvertor(request.getBucketName()).transCannedAcl(request.getCannedACL().trim()));
}
Map headers = new HashMap<>();
headers.put(CommonHeaders.CONTENT_TYPE, Mimetypes.MIMETYPE_XML);
boolean isExtraAclPutRequired = !prepareRESTHeaderAclObject(request.getBucketName(), headers, request.getAcl());
if (isExtraAclPutRequired) {
String xml = request.getAcl() == null ? ""
: this.getIConvertor(request.getBucketName()).transAccessControlList(request.getAcl(), false);
headers.put(CommonHeaders.CONTENT_LENGTH, String.valueOf(xml.length()));
headers.put(CommonHeaders.CONTENT_MD5, ServiceUtils.computeMD5(xml));
entity = createRequestBody(Mimetypes.MIMETYPE_XML, xml);
}
transRequestPaymentHeaders(request, headers, this.getIHeaders(request.getBucketName()));
NewTransResult result = transObjectRequest(request);
result.setParams(requestParams);
result.setHeaders(headers);
result.setBody(entity);
Response response = performRequest(result);
return build(response);
}
protected AccessControlList getObjectAclImpl(GetObjectAclRequest getObjectAclRequest) throws ServiceException {
Map requestParameters = new HashMap<>();
requestParameters.put(SpecialParamEnum.ACL.getOriginalStringCode(), "");
if (ServiceUtils.isValid(getObjectAclRequest.getVersionId())) {
requestParameters.put(ObsRequestParams.VERSION_ID, getObjectAclRequest.getVersionId().trim());
}
Response httpResponse = performRestGet(getObjectAclRequest.getBucketName(), getObjectAclRequest.getObjectKey(),
requestParameters, transRequestPaymentHeaders(getObjectAclRequest, null,
this.getIHeaders(getObjectAclRequest.getBucketName())), getObjectAclRequest.getUserHeaders());
this.verifyResponseContentType(httpResponse);
AccessControlList ret = getXmlResponseSaxParser().parse(new HttpMethodReleaseInputStream(httpResponse),
XmlResponsesSaxParser.AccessControlListHandler.class, false).getAccessControlList();
setHeadersAndStatus(ret, httpResponse);
return ret;
}
protected String getObjectUrl(String bucketName, String objectKey, boolean isIgnorePort) {
boolean pathStyle = this.isPathStyle();
boolean https = this.getHttpsOnly();
boolean isCname = this.isCname();
String port = "";
if (!isIgnorePort) {
port = ":" + (https ? this.getHttpsPort() : this.getHttpPort());
}
return (https ? "https://" : "http://") + (pathStyle || isCname ? "" : bucketName + ".")
+ this.getEndpoint() + port + "/" + (pathStyle ? bucketName + "/" : "")
+ RestUtils.uriEncode(objectKey, false);
}
protected ObsFSAttribute getObsFSAttributeFromResponse(String bucketName, Response response, boolean needDecode) {
Date lastModifiedDate = null;
String lastModified = response.header(CommonHeaders.LAST_MODIFIED);
if (lastModified != null) {
try {
lastModifiedDate = ServiceUtils.parseRfc822Date(lastModified);
} catch (ParseException e) {
if (log.isWarnEnabled()) {
log.warn("Response last-modified is not well-format", e);
}
}
}
ObsFSAttribute objMetadata = new ObsFSAttribute();
objMetadata.setLastModified(lastModifiedDate);
objMetadata.setContentEncoding(response.header(CommonHeaders.CONTENT_ENCODING));
objMetadata.setContentType(response.header(CommonHeaders.CONTENT_TYPE));
objMetadata.setContentDisposition(response.header(CommonHeaders.CONTENT_DISPOSITION));
objMetadata.setContentLanguage(response.header(CommonHeaders.CONTENT_LANGUAGE));
objMetadata.setCacheControl(response.header(CommonHeaders.CACHE_CONTROL));
objMetadata.setExpires(response.header(CommonHeaders.EXPIRES));
String crc64 = response.header(OBS_HEADER_PREFIX + CommonHeaders.HASH_CRC64ECMA);
if (crc64 == null || crc64.isEmpty()) {
crc64 = response.header(V2_HEADER_PREFIX + CommonHeaders.HASH_CRC64ECMA);
}
objMetadata.setCrc64(crc64);
String contentLength = response.header(CommonHeaders.CONTENT_LENGTH);
if (contentLength != null) {
try {
objMetadata.setContentLength(Long.parseLong(contentLength));
} catch (NumberFormatException e) {
if (log.isWarnEnabled()) {
log.warn("Response content-length is not well-format", e);
}
}
}
String fsMode;
if ((fsMode = response.header(this.getIHeaders(bucketName).fsModeHeader())) != null) {
objMetadata.setMode(Integer.parseInt(fsMode));
}
objMetadata.setWebSiteRedirectLocation(response.header(this.getIHeaders(bucketName)
.websiteRedirectLocationHeader()));
objMetadata.setObjectStorageClass(
StorageClassEnum.getValueFromCode(response.header(this.getIHeaders(bucketName).storageClassHeader())));
String etag = response.header(CommonHeaders.ETAG);
objMetadata.setEtag(etag);
if (etag != null && !etag.contains("-")) {
String md5 = etag;
if (md5.startsWith("\"")) {
md5 = md5.substring(1);
}
if (md5.endsWith("\"")) {
md5 = md5.substring(0, md5.length() - 1);
}
try {
objMetadata.setContentMd5(ServiceUtils.toBase64(ServiceUtils.fromHex(md5)));
} catch (Exception e) {
if (log.isDebugEnabled()) {
log.debug(e.getMessage(), e);
}
}
}
objMetadata.setAppendable("Appendable".equals(response.header(this.getIHeaders(bucketName)
.objectTypeHeader())));
String nextPosition = response.header(this.getIHeaders(bucketName).nextPositionHeader(), "-1");
objMetadata.setNextPosition(Long.parseLong(nextPosition));
if (objMetadata.getNextPosition() == -1L) {
objMetadata.setNextPosition(Long.parseLong(response.header(Constants.CommonHeaders.CONTENT_LENGTH, "-1")));
}
setHeadersAndStatus(objMetadata, response, needDecode);
objMetadata.setUserMetadata(ServiceUtils.cleanUserMetadata(objMetadata.getOriginalHeaders(), needDecode));
return objMetadata;
}
}