123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447 |
- package com.fdkankan.fyun.s3;
- import cn.hutool.core.collection.CollUtil;
- import com.amazonaws.HttpMethod;
- import com.amazonaws.services.s3.AmazonS3;
- import com.amazonaws.services.s3.model.*;
- import com.fdkankan.fyun.constant.FYunTypeEnum;
- import com.fdkankan.fyun.face.AbstractFYunFileService;
- import org.apache.commons.lang3.StringUtils;
- import org.slf4j.Logger;
- import org.slf4j.LoggerFactory;
- import org.springframework.beans.factory.annotation.Autowired;
- import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
- import org.springframework.stereotype.Component;
- import org.springframework.util.ObjectUtils;
- import java.io.*;
- import java.net.URL;
- import java.util.*;
- import java.util.stream.Collectors;
- @Component
- @ConditionalOnProperty(name = "fyun.type",havingValue = "aws")
- public class S3FileService extends AbstractFYunFileService {
- private Logger log = LoggerFactory.getLogger(this.getClass().getName());
- @Autowired
- private AmazonS3 s3;
- @Override
- public String uploadFile(String bucket, byte[] data, String remoteFilePath){
- try {
- ObjectMetadata metadata = new ObjectMetadata();
- PutObjectRequest request = new PutObjectRequest(bucket, remoteFilePath, new ByteArrayInputStream(data), metadata);
- request.withCannedAcl(CannedAccessControlList.PublicRead);
- s3.putObject(request);
- } catch (Exception e) {
- log.error("s3上传文件失败", e);
- }
- return null;
- }
- @Override
- public String uploadFile(String bucket, String filePath, String remoteFilePath){
- return uploadFile(bucket, filePath, remoteFilePath,null);
- }
- @Override
- public String uploadFile(String bucket, InputStream inputStream, String remoteFilePath) {
- try {
- s3.putObject(bucket, remoteFilePath, inputStream, null);
- } catch (Exception e) {
- log.error("文件流上传失败,目标路径:{}", remoteFilePath);
- e.printStackTrace();
- }
- return null;
- }
- @Override
- public String uploadFile(String bucket, String filePath, String remoteFilePath,Map<String, String> headers){
- try {
- File file = new File(filePath);
- if (!file.exists()) {
- log.error("要上传的文件不存在:" + filePath);
- return null;
- }
- // 设置文件并设置公读
- ObjectMetadata metadata = new ObjectMetadata();
- if (filePath.contains(".jpg")) {
- metadata.setContentType("image/jpeg");
- }
- if (filePath.contains(".png")) {
- metadata.setContentType("image/png");
- }
- if(org.apache.commons.lang3.ObjectUtils.isNotEmpty(headers)){
- for (Map.Entry<String, String> header : headers.entrySet()) {
- metadata.setHeader(header.getKey(),header.getValue());
- }
- }
- PutObjectRequest request = new PutObjectRequest(bucket, remoteFilePath, file);
- request.withCannedAcl(CannedAccessControlList.PublicRead);
- request.withMetadata(metadata);
- // 上传文件
- PutObjectResult putObjectResult = s3.putObject(request);
- if (StringUtils.isNotEmpty(putObjectResult.getETag())) {
- log.info("s3上传文件成功:" + remoteFilePath);
- }
- } catch (Exception e) {
- log.error("文件上传失败:{}",filePath);
- e.printStackTrace();
- }
- return null;
- }
- @Override
- public String uploadFileByCommand(String bucket, String filePath, String remoteFilePath){
- try {
- String optType = new File(filePath).isDirectory() ? "folder" : "file";
- String command = String.format(fYunConstants.UPLOAD_SH, bucket, filePath, remoteFilePath, FYunTypeEnum.AWS.code(), optType);
- log.info("开始上传文件, ossPath:{}, srcPath:{}", remoteFilePath, filePath);
- callshell(command);
- } catch (Exception e) {
- log.error("上传文件失败, ossPath:{}, srcPath:{}", remoteFilePath, filePath);
- e.printStackTrace();
- }
- return null;
- }
- @Override
- public void downloadFileByCommand(String bucket, String filePath, String remoteFilePath) {
- try {
- String optType = remoteFilePath.contains(".") ? "file" : "folder";
- String command = String.format(fYunConstants.DOWNLOAD_SH, bucket, remoteFilePath, filePath, FYunTypeEnum.AWS.code(), optType);
- log.info("开始下载文件, ossPath:{}, srcPath:{}", remoteFilePath, filePath);
- callshell(command);
- } catch (Exception e) {
- log.error("上传文件失败, ossPath:{}, srcPath:{}", remoteFilePath, filePath);
- e.printStackTrace();
- }
- }
- @Override
- public void deleteFile(String bucket, String remoteFilePath){
- if (remoteFilePath.startsWith("/")) {
- remoteFilePath = remoteFilePath.substring(1);
- }
- try {
- s3.deleteObject(bucket, remoteFilePath);
- } catch (Exception e) {
- log.error("s3删除文件失败,key=" + remoteFilePath, e);
- }
- }
- @Override
- public void deleteFolder(String bucket, String remoteFolderPath){
- try {
- if (!remoteFolderPath.endsWith("/")) {
- remoteFolderPath = remoteFolderPath + "/";
- }
- log.info("开始删除文件夹:{}", remoteFolderPath);
- int maxKeys = 1000;
- String nextMaker = null;
- ListObjectsRequest listObjectsRequest = new ListObjectsRequest();
- listObjectsRequest.setBucketName(bucket);
- listObjectsRequest.setPrefix(remoteFolderPath);
- listObjectsRequest.setMaxKeys(maxKeys);
- ObjectListing objectListing;
- DeleteObjectsRequest multiObjectDeleteRequest = new DeleteObjectsRequest(bucket).withQuiet(false);
- do {
- listObjectsRequest.setMarker(nextMaker);
- objectListing = s3.listObjects(listObjectsRequest);
- List<S3ObjectSummary> objectSummaries = objectListing.getObjectSummaries();
- List<DeleteObjectsRequest.KeyVersion> keys = objectSummaries.stream().map(summary -> new DeleteObjectsRequest.KeyVersion(summary.getKey())).collect(Collectors.toList());
- if (!ObjectUtils.isEmpty(keys)) {
- multiObjectDeleteRequest.setKeys(keys);
- s3.deleteObjects(multiObjectDeleteRequest);
- }
- nextMaker = objectListing.getNextMarker();
- } while (objectListing.isTruncated());
- } catch (Exception e) {
- log.error("删除aws文件失败,path=" + remoteFolderPath, e);
- }
- }
- @Override
- public void uploadMulFiles(String bucket, Map<String, String> filepaths){
- try {
- for (Map.Entry<String, String> entry : filepaths.entrySet()) {
- uploadFile(bucket, entry.getKey(), entry.getValue(),null);
- }
- } catch (Exception e) {
- log.error("OSS批量上传文件失败!");
- }
- }
- @Override
- public List<String> listRemoteFiles(String bucket, String sourcePath) {
- List<String> keyList = new ArrayList<>();
- try {
- boolean flag = true;
- String nextMaker = null;
- ListObjectsRequest listObjectsRequest = new ListObjectsRequest();
- listObjectsRequest.setBucketName(bucket);
- listObjectsRequest.setPrefix(sourcePath);
- listObjectsRequest.setMaxKeys(200);
- do {
- listObjectsRequest.setMarker(nextMaker);
- ObjectListing objectListing = s3.listObjects(listObjectsRequest);
- List<S3ObjectSummary> objectSummaries = objectListing.getObjectSummaries();
- List<String> collect = objectSummaries.stream().map(S3ObjectSummary::getKey).collect(Collectors.toList());
- if (CollUtil.isNotEmpty(collect)) {
- keyList.addAll(collect);
- }
- nextMaker = objectListing.getNextMarker();
- flag = objectListing.isTruncated();
- } while (flag);
- } catch (Exception e) {
- log.error("获取文件列表失败,path=" + sourcePath, e);
- e.printStackTrace();
- }
- return keyList;
- }
- @Override
- public void copyFileBetweenBucket(String sourceBucketName, String sourcePath, String targetBucketName, String targetPath){
- try {
- List<String> files = listRemoteFiles(sourceBucketName, sourcePath);
- if (ObjectUtils.isEmpty(files)) {
- return;
- }
- files.stream().forEach(file -> {
- CopyObjectRequest request = new CopyObjectRequest(sourceBucketName, file, targetBucketName, file.replace(sourcePath, targetPath));
- request.withCannedAccessControlList(CannedAccessControlList.PublicRead);
- s3.copyObject(request);
- });
- } catch (Exception e) {
- log.error("列举文件目录失败,key=" + sourcePath);
- }
- }
- @Override
- public void copyFilesBetweenBucket(String sourceBucketName, String targetBucketName, Map<String, String> pathMap){
- if (ObjectUtils.isEmpty(pathMap)) {
- return;
- }
- try {
- for (Map.Entry<String, String> entry : pathMap.entrySet()) {
- copyFileBetweenBucket(sourceBucketName, entry.getKey(), targetBucketName, entry.getValue());
- }
- } catch (Exception e) {
- log.error("批量复制文件失败!");
- }
- }
- @Override
- public String getFileContent(String bucketName, String remoteFilePath){
- try {
- GetObjectRequest request = new GetObjectRequest(bucketName,remoteFilePath);
- S3Object object = s3.getObject(request);
- S3ObjectInputStream inputStream = object.getObjectContent();
- StringBuilder content = new StringBuilder();
- try(BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream))){
- while (true) {
- String line = reader.readLine();
- if (line == null) break;
- content.append(line);
- }
- } catch (IOException e) {
- log.error("读取aws文件流失败", e);
- }
- object.close();
- return content.toString();
- } catch (Exception e) {
- log.error("获取文件内容失败:{}", remoteFilePath);
- return null;
- }
- }
- @Override
- public boolean fileExist(String bucket, String objectName) {
- try {
- return s3.doesObjectExist(bucket, objectName);
- } catch (Exception e) {
- log.error("判断文件是否存在失败:{}", objectName);
- return false;
- }
- }
- @Override
- public void downloadFile(String bucket, String remoteFilePath, String localPath) {
- try {
- File localFile = new File(localPath);
- if (!localFile.getParentFile().exists()) {
- localFile.getParentFile().mkdirs();
- }
- if(localFile.isDirectory()){
- String fileName = remoteFilePath.substring(remoteFilePath.lastIndexOf("/")+1);
- log.info("未配置文件名,使用默认文件名:{}",fileName);
- localPath = localPath.concat(File.separator).concat(fileName);
- }
- GetObjectRequest request = new GetObjectRequest(bucket, remoteFilePath);
- s3.getObject(request,new File(localPath));
- } catch (Throwable throwable) {
- log.error("文件下载失败:{}", remoteFilePath);
- throwable.printStackTrace();
- }
- }
- @Override
- public URL getPresignedUrl(String bucket,String url) {
- java.util.Date expiration = new java.util.Date();
- long expTimeMillis = expiration.getTime();
- expTimeMillis += 1000 * 60 * 60 * 8;
- expiration.setTime(expTimeMillis);
- GeneratePresignedUrlRequest generatePresignedUrlRequest = new GeneratePresignedUrlRequest(bucket, url)
- .withMethod(HttpMethod.PUT).withExpiration(expiration);
- return s3.generatePresignedUrl(generatePresignedUrlRequest);
- }
- @Override
- public long getSubFileNums(String bucket, String url) {
- long totalFileNums = 0;
- try {
- boolean flag = true;
- String nextMaker = null;
- ListObjectsRequest listObjectsRequest = new ListObjectsRequest();
- listObjectsRequest.setBucketName(bucket);
- listObjectsRequest.setPrefix(url);
- listObjectsRequest.setMaxKeys(200);
- do {
- listObjectsRequest.setMarker(nextMaker);
- ObjectListing objectListing = s3.listObjects(listObjectsRequest);
- List<S3ObjectSummary> objectSummaries = objectListing.getObjectSummaries();
- List<String> collect = objectSummaries.stream().map(S3ObjectSummary::getKey).collect(Collectors.toList());
- totalFileNums = totalFileNums + collect.size();
- nextMaker = objectListing.getNextMarker();
- flag = objectListing.isTruncated();
- } while (flag);
- } catch (Exception e) {
- log.error("获取文件数量失败,path=" + url, e);
- e.printStackTrace();
- }
- return totalFileNums;
- }
- @Override
- public void restoreFolder(String bucket, String folderName, Integer priority) {
- List<String> objectList = this.listRemoteFiles(bucket, folderName);
- if(CollUtil.isEmpty(objectList)){
- return;
- }
- objectList.parallelStream().forEach(objectName -> {
- this.restoreFile(bucket, objectName, priority);
- });
- }
- @Override
- public void restoreFile(String bucket, String objectName, Integer priority){
- // ObjectMetadata objectMetadata = ossClient.getObjectMetadata(bucket, objectName);
- //
- // // 校验Object是否为归档类型Object。
- // StorageClass storageClass = objectMetadata.getObjectStorageClass();
- // if (storageClass == StorageClass.ColdArchive) {
- // // 设置解冻冷归档Object的优先级。
- // // RestoreTier.RESTORE_TIER_EXPEDITED 表示1小时内完成解冻。
- // // RestoreTier.RESTORE_TIER_STANDARD 表示2~5小时内完成解冻。
- // // RestoreTier.RESTORE_TIER_BULK 表示5~12小时内完成解冻。
- // RestoreTier restoreTier = null;
- // switch (priority){
- // case 1 :
- // restoreTier = RestoreTier.RESTORE_TIER_EXPEDITED;
- // break;
- // case 2 :
- // restoreTier = RestoreTier.RESTORE_TIER_STANDARD;
- // break;
- // default:
- // restoreTier = RestoreTier.RESTORE_TIER_BULK;
- // }
- // RestoreJobParameters jobParameters = new RestoreJobParameters(restoreTier);
- // // 配置解冻参数,以设置5小时内解冻完成,解冻状态保持2天为例。
- // // 第一个参数表示保持解冻状态的天数,默认是1天,此参数适用于解冻Archive(归档)与ColdArchive(冷归档)类型Object。
- // // 第二个参数jobParameters表示解冻优先级,只适用于解冻ColdArchive类型Object。
- // RestoreConfiguration configuration = new RestoreConfiguration(1, jobParameters);
- // //开始解冻
- // ossClient.restoreObject(bucket, objectName, configuration);
- //// // 等待解冻完成。
- //// do {
- //// try {
- //// Thread.sleep(1000);
- //// } catch (InterruptedException e) {
- //// e.printStackTrace();
- //// }
- //// objectMetadata = ossClient.getObjectMetadata(bucket, objectName);
- //// } while (!objectMetadata.isRestoreCompleted());
- // }
- }
- @Override
- public Boolean checkStore(String bucket, String url) {
- ObjectMetadata objectMetadata = s3.getObjectMetadata(bucket, url);
- return !isRestoreCompleted(objectMetadata);
- }
- private boolean isRestoreCompleted(ObjectMetadata objectMetadata){
- Date restoreExpirationTime = objectMetadata.getRestoreExpirationTime();
- if(Objects.nonNull(restoreExpirationTime) && restoreExpirationTime.after(Calendar.getInstance().getTime())){
- return true;
- }
- return false;
- }
- @Override
- public void restoreFolder(String bucket, String url) {
- ObjectMetadata objectMetadata ;
- List<String> objectList = this.listRemoteFiles(bucket, url);
- if(CollUtil.isEmpty(objectList)){
- return;
- }
- for (String objectName : objectList) {
- objectMetadata = s3.getObjectMetadata(bucket, objectName);
- // 校验Object是否为归档类型Object。
- StorageClass storageClass = objectMetadata.getArchiveStatus();
- if (storageClass == StorageClass.) {
- // 解冻Object。
- RestoreObjectRequest requestRestore = new RestoreObjectRequest(bucket, url, 1);
- s3.restoreObjectV2(requestRestore);
- }
- }
- }
- @Override
- public Integer getRestoreFolderProcess(String bucket, String url) {
- return null;
- }
- @Override
- public Long getSpace(String bucket, String key) {
- List<String> keyList = new ArrayList<>();
- Long total = 0L;
- boolean flag = true;
- String nextMaker = null;
- ListObjectsRequest listObjectsRequest = new ListObjectsRequest();
- listObjectsRequest.setBucketName(bucket);
- listObjectsRequest.setPrefix(key);
- listObjectsRequest.setMaxKeys(200);
- do {
- listObjectsRequest.setMarker(nextMaker);
- ObjectListing objectListing = s3.listObjects(listObjectsRequest);
- List<S3ObjectSummary> objectSummaries = objectListing.getObjectSummaries();
- Long space = objectSummaries.stream().mapToLong(S3ObjectSummary::getSize).sum();
- total += space;
- nextMaker = objectListing.getNextMarker();
- flag = objectListing.isTruncated();
- } while (flag);
- return total;
- }
- }
|