1

我正在尝试将一个大文件上传到使用令牌的服务器,并且令牌在 10 分钟后过期,所以如果我上传一个小文件,它将起作用,因此如果文件大,我会遇到一些问题并会尝试在访问被拒绝时永久上传

所以我需要刷新BasicAWSCredentials中的令牌,而不是用于AWSStaticCredentialsProvider因此我不确定我该怎么做,请帮助=)

值得一提的是,我们使用本地服务器(不是亚马逊云)提供令牌,为了方便我们使用亚马逊的代码。

这是我的代码:

public void uploadMultipart(File file) throws Exception {
    //this method will give you a initial token for a given user, 
    //than calculates when a new token is needed and will refresh it just when necessary

    String token = getUsetToken();
    String existingBucketName = myTenant.toLowerCase() + ".package.upload";
    String endPoint = urlAPI + "s3/buckets/";
    String strSize = FileUtils.byteCountToDisplaySize(FileUtils.sizeOf(file));
    System.out.println("File size: " + strSize);

    AwsClientBuilder.EndpointConfiguration endpointConfiguration = new AwsClientBuilder.EndpointConfiguration(endPoint, null);//note: Region has to be null
    //AWSCredentialsProvider        
    BasicAWSCredentials sessionCredentials = new BasicAWSCredentials(token, "NOT_USED");//secretKey should be set to NOT_USED

    AmazonS3 s3 = AmazonS3ClientBuilder
            .standard()
            .withCredentials(new AWSStaticCredentialsProvider(sessionCredentials))
            .withEndpointConfiguration(endpointConfiguration)
            .enablePathStyleAccess()
            .build();

    int maxUploadThreads = 5;
    TransferManager tm = TransferManagerBuilder
            .standard()
            .withS3Client(s3)
            .withMultipartUploadThreshold((long) (5 * 1024 * 1024))
            .withExecutorFactory(() -> Executors.newFixedThreadPool(maxUploadThreads))
            .build();

    PutObjectRequest request = new PutObjectRequest(existingBucketName, file.getName(), file);
    //request.putCustomRequestHeader("Access-Token", token);
    ProgressListener progressListener = progressEvent -> System.out.println("Transferred bytes: " + progressEvent.getBytesTransferred());
    request.setGeneralProgressListener(progressListener);
    Upload upload = tm.upload(request);

    LocalDateTime uploadStartedAt = LocalDateTime.now();
    log.info("Starting upload at: " + uploadStartedAt);

    try {
        upload.waitForCompletion();
        //upload.waitForUploadResult();
        log.info("Upload completed. " + strSize);

    } catch (Exception e) {//AmazonClientException
        log.error("Error occurred while uploading file - " + strSize);
        e.printStackTrace();
    }
}
4

2 回答 2

2

找到解决方案!

我找到了一种让它工作的方法,老实说我对结果很满意,我已经用大文件(50gd.zip)做了很多测试,并且在每个场景中都运行得很好

我的解决方案是,删除该行: BasicAWSCredentials sessionCredentials = new BasicAWSCredentials(token, "NOT_USED");

AWSCredentials是一个接口,因此我们可以用动态的东西覆盖它,令牌过期并需要新令牌的逻辑保存在getToken()方法中,这意味着您可以每次调用都不会造成伤害

AWSCredentials sessionCredentials = new AWSCredentials() {
    @Override
    public String getAWSAccessKeyId() {
        try {
            return getToken(); //getToken() method return a string 
        } catch (Exception e) {
            return null;
        }
    }

    @Override
    public String getAWSSecretKey() {
        return "NOT_USED";
    }
};
于 2018-08-16T09:07:14.507 回答
1

上传文件(或多部分文件的一部分)时,您使用的凭据必须持续足够长的时间才能完成上传。您无法刷新凭证,因为没有方法可以更新 AWS S3,您正在为已签名的请求使用新凭证。

您可以将上传分成更小的文件,以便更快地上传。然后只上传 X 部分。刷新您的凭据并上传 Y 部分。重复直到上传所有部分。然后你需要通过组合部件来完成(这是一个单独的命令)。这不是一个完美的解决方案,因为无法准确控制传输速度,这意味着您必须编写自己的上传代码(这并不难)。

于 2018-08-14T04:43:38.353 回答