首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >Alpakka S3连接问题

Alpakka S3连接问题
EN

Stack Overflow用户
提问于 2020-05-21 15:30:29
回答 2查看 1K关注 0票数 2

为了存储文件,我试图使用Alpakka S3连接到minio实例,但是自从我将库版本从1.1.2升级到2.0.0之后,我遇到了一些问题。

下面是一个简单的服务类,它只包含两个尝试创建桶的方法。我尝试了两种方法,首先从本地配置文件(在我的例子中是application.conf)加载alpakka设置,然后通过S3Ext直接创建设置。

这两种方法都失败了,我对这个问题不太确定。关于错误,似乎设置没有正确加载,但我不知道我在这里做错了什么。

我用的是:

  • 游戏框架2.8.1
  • 斯卡拉2.13.2
  • akka-stream-alpakka-s3 2.0.0

以下是服务类:

代码语言:javascript
复制
package services

import akka.actor.ActorSystem
import akka.stream.alpakka.s3._
import akka.stream.alpakka.s3.scaladsl.S3
import akka.stream.scaladsl.Sink
import akka.stream.{Attributes, Materializer}
import javax.inject.{Inject, Singleton}
import software.amazon.awssdk.auth.credentials.{AwsBasicCredentials, AwsCredentials, AwsCredentialsProvider}
import software.amazon.awssdk.regions.Region
import software.amazon.awssdk.regions.providers.AwsRegionProvider

import scala.concurrent.ExecutionContext.Implicits.global
import scala.concurrent.Future

@Singleton
class AlpakkaS3PlaygroundService @Inject()(
    materializer: Materializer,
    system: ActorSystem,
) {

  def makeBucket(bucketName: String): Future[String] = {
    S3.makeBucket(bucketName)(materializer) map { _ =>
      "bucket created"
    }
  }

  def makeBucket2(bucketName: String): Future[String] = {

    val s3Host      = "http://localhost:9000"
    val s3AccessKey = "access_key"
    val s3SecretKey = "secret_key"
    val s3Region    = "eu-central-1"

    val credentialsProvider = new AwsCredentialsProvider {
      override def resolveCredentials(): AwsCredentials = AwsBasicCredentials.create(s3AccessKey, s3SecretKey)
    }

    val regionProvider = new AwsRegionProvider {
      override def getRegion: Region = Region.of(s3Region)
    }

    val settings: S3Settings = S3Ext(system).settings
      .withEndpointUrl(s3Host)
      .withBufferType(MemoryBufferType)
      .withCredentialsProvider(credentialsProvider)
      .withListBucketApiVersion(ApiVersion.ListBucketVersion2)
      .withS3RegionProvider(regionProvider)

    val attributes: Attributes = S3Attributes.settings(settings)

    S3.makeBucketSource(bucketName)
      .withAttributes(attributes)
      .runWith(Sink.head)(materializer) map { _ =>
      "bucket created"
    }
  }
}

application.conf中的配置如下所示:

代码语言:javascript
复制
akka.stream.alpakka.s3 {
  aws {
    credentials {
      provider = static
      access-key-id = "access_key"
      secret-access-key = "secret_key"
    }
    region {
      provider = static
      default-region = "eu-central-1"
    }
  }
  endpoint-url = "http://localhost:9000"
}

如果使用服务的第一个方法(makeBucket(...)),我会看到以下错误:

代码语言:javascript
复制
SdkClientException: Unable to load region from any of the providers in the chain software.amazon.awssdk.regions.providers.DefaultAwsRegionProviderChain@34cb16dc: 
[software.amazon.awssdk.regions.providers.SystemSettingsRegionProvider@804e08b: Unable to load region from system settings. Region must be specified either via environment variable (AWS_REGION) or  system property (aws.region)., software.amazon.awssdk.regions.providers.AwsProfileRegionProvider@4d5f4b4d: No region provided in profile: default, software.amazon.awssdk.regions.providers.InstanceProfileRegionProvider@557feb58: Unable to contact EC2 metadata service.]

错误消息非常精确,我知道哪里出了问题,但我只是不知道该怎么做,因为我指定了文档中列出的设置。有什么想法吗?

在服务的第二种方法(makeBucket2(...))中,我试图显式地设置S3设置,但这似乎也不起作用。错误如下所示:

代码语言:javascript
复制
play.api.http.HttpErrorHandlerExceptions$$anon$1: Execution exception[[S3Exception: 404 page not found
]]
    at play.api.http.HttpErrorHandlerExceptions$.throwableToUsefulException(HttpErrorHandler.scala:335)
    at play.api.http.DefaultHttpErrorHandler.onServerError(HttpErrorHandler.scala:253)
    at play.core.server.AkkaHttpServer$$anonfun$2.applyOrElse(AkkaHttpServer.scala:424)
    at play.core.server.AkkaHttpServer$$anonfun$2.applyOrElse(AkkaHttpServer.scala:420)
    at scala.concurrent.impl.Promise$Transformation.run(Promise.scala:453)
    at akka.dispatch.BatchingExecutor$AbstractBatch.processBatch(BatchingExecutor.scala:55)
    at akka.dispatch.BatchingExecutor$BlockableBatch.$anonfun$run$1(BatchingExecutor.scala:92)
    at scala.runtime.java8.JFunction0$mcV$sp.apply(JFunction0$mcV$sp.scala:18)
    at scala.concurrent.BlockContext$.withBlockContext(BlockContext.scala:94)
    at akka.dispatch.BatchingExecutor$BlockableBatch.run(BatchingExecutor.scala:92)
    at akka.dispatch.TaskInvocation.run(AbstractDispatcher.scala:47)
    at akka.dispatch.ForkJoinExecutorConfigurator$AkkaForkJoinTask.exec(ForkJoinExecutorConfigurator.scala:47)
    at java.base/java.util.concurrent.ForkJoinTask.doExec(ForkJoinTask.java:290)
    at java.base/java.util.concurrent.ForkJoinPool$WorkQueue.topLevelExec(ForkJoinPool.java:1020)
    at java.base/java.util.concurrent.ForkJoinPool.scan(ForkJoinPool.java:1656)
    at java.base/java.util.concurrent.ForkJoinPool.runWorker(ForkJoinPool.java:1594)
    at java.base/java.util.concurrent.ForkJoinWorkerThread.run(ForkJoinWorkerThread.java:177)
Caused by: akka.stream.alpakka.s3.S3Exception: 404 page not found

在这里,似乎根本没有考虑到定义的设置,因为似乎找不到服务。这实际上是我在以前版本的软件中使用的方法,我使用的是akka-stream-alpakka-s3版本1.1.2,它按照预期工作。

当然,我不仅想使用Alpakka S3来创建桶,而且为了这个展示和概述我的问题,我只使用这个示例来保持简单。我想,如果这个问题解决了,alpakka提供的所有其他方法都会奏效。

我确实对文档进行了多次修改,但我仍然无法解决这个问题,所以我希望这里有人能帮助我。

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2020-05-22 14:47:09

我在光明论坛这里上得到了帮助。

通过设置以下参数解决了这个问题:

代码语言:javascript
复制
alpakka.s3.path-style-access = true

因为文档中说,这个值将被废弃,所以我没有考虑指定它。

在我最初的文章中,我概述了两种设置params的方法,一种是通过application.conf,另一种是通过S3Ext编程。第一种方法通过设置上面所示的值来工作,第二种方法如下所示:

代码语言:javascript
复制
val settings: S3Settings = S3Ext(system).settings
      .withEndpointUrl(s3Host)
      .withBufferType(MemoryBufferType)
      .withCredentialsProvider(credentialsProvider)
      .withListBucketApiVersion(ApiVersion.ListBucketVersion2)
      .withS3RegionProvider(regionProvider)
      .withPathStyleAccess(true)

这里的最后一行是至关重要的,尽管我收到了一个反对警告。

但最终,这解决了这个问题。

票数 1
EN

Stack Overflow用户

发布于 2020-05-21 17:59:13

至少在2.0.0时,Alpakka S3的配置路径是alpakka.s3而不是akka.stream.alpakka.s3

代码语言:javascript
复制
alpakka.s3 {
  aws {
    credentials {
      provider = static
      access-key-id = "access_key"
      secret-access-key = "secret_key"
    }
    region {
      provider = static
      default-region = "eu-central-1"
    }
  }
  endpoint-url = "http://localhost:9000"
}
票数 2
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/61938052

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档