是否可以在Java中使用Play的Iteratee
?我无法在 Java 中使用Iteratee
找到任何示例或doco,只有Scala。我猜测让Iteratees使用PLay API在Java中工作是一个更混乱的代码明智(很多anon Funtion1<?,>
s)......
如果可能的话,我想创建一个App控制器,它可以接受通过HTTPs chunked transfer encoding上传的多部分文件上传,并将这些消息块下游解析到S3存储。关于如何在Java中解决这个问题的任何想法?
干杯。
答案 0 :(得分:4)
Java SDK包含用于执行异步上载的类TransferManager。 它包含一个自己的可配置ThreadPool。
用Java编写的Iteratees可能能够将上传文件的字节直接推送到S3,但代码看起来很难并且难以配置。对于很多用例来说,将文件从浏览器流式传输到临时文件(因此它不完全在内存中)然后将其流式传输到S3就足够了。
我在Github上创建了一个示例项目,如何做到这一点:
package controllers;
import com.amazonaws.auth.AWSCredentials;
import com.amazonaws.auth.BasicAWSCredentials;
import com.amazonaws.event.ProgressEvent;
import com.amazonaws.event.ProgressListener;
import com.amazonaws.services.s3.transfer.TransferManager;
import com.amazonaws.services.s3.transfer.Upload;
import com.amazonaws.services.s3.transfer.model.UploadResult;
import play.*;
import play.libs.F.Function;
import play.libs.F.Promise;
import play.mvc.*;
import views.html.index;
import scala.concurrent.Promise$;
public class Application extends Controller {
//don't forget to tm.shutdownNow() on application termination
//you can configure a Thread pool http://docs.aws.amazon.com/AWSJavaSDK/latest/javadoc/com/amazonaws/services/s3/transfer/TransferManager.html#TransferManager(com.amazonaws.services.s3.AmazonS3, java.util.concurrent.ThreadPoolExecutor)
private static TransferManager tm;
private static final String BUCKET_NAME = "your-bucket";
//this is bad style, use a plugin!
static {
final String accessKey = System.getenv("AWS_ACCESS");
final String secretKey = System.getenv("AWS_SECRET");
final AWSCredentials credentials = new BasicAWSCredentials(accessKey, secretKey);
tm = new TransferManager(credentials);
}
/** shows a form to upload a file */
public static Result index() {
return ok(index.render("Your new application is ready."));
}
/** uploads a file to Amazon S3. */
public static Promise<Result> upload() {
final Http.MultipartFormData.FilePart meta = request().body().asMultipartFormData().getFile("picture");
final String key = meta.getFilename();
final Upload upload = tm.upload(BUCKET_NAME, key, meta.getFile());
Logger.info("start upload " + meta.getFilename());
return asPromise(meta.getFilename(), upload).map(new Function<UploadResult, Result>() {
@Override
public Result apply(UploadResult uploadResult) throws Throwable {
Logger.info("finished " + meta.getFilename());
return ok(asString(uploadResult));
}
});
}
private static String asString(UploadResult result) {
return "UploadResult{bucketName=" + result.getBucketName() + ", key=" + result.getKey() + ", version=" + result.getVersionId() + ", ETag=" + result.getETag() + "}";
}
private static Promise<UploadResult> asPromise(final String filename, final Upload upload) {
final scala.concurrent.Promise<UploadResult> scalaPromise = Promise$.MODULE$.apply();
upload.addProgressListener(new ProgressListener() {
@Override
public void progressChanged(ProgressEvent progressEvent) {
if (progressEvent.getEventCode() == ProgressEvent.CANCELED_EVENT_CODE) {
scalaPromise.failure(new RuntimeException("canceled " + filename));
} else if (progressEvent.getEventCode() == ProgressEvent.FAILED_EVENT_CODE) {
scalaPromise.failure(new RuntimeException("failed " + filename));
} else if(progressEvent.getEventCode() == ProgressEvent.COMPLETED_EVENT_CODE) {
Logger.info("done " + filename);
try {
scalaPromise.success(upload.waitForUploadResult());
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
}
}
});
return Promise.wrap(scalaPromise.future());
}
}
用于在不同浏览器窗口中上传两个文件的示例日志输出:
[info] play - play-internal-execution-context-1 Application started (Dev)
[info] application - play-akka.actor.default-dispatcher-3 start upload file5.ext
[info] application - play-akka.actor.default-dispatcher-2 start upload file1.ext
[info] application - java-sdk-progress-listener-callback-thread done file1.ext
[info] application - play-akka.actor.default-dispatcher-5 finished file1.ext
[info] application - java-sdk-progress-listener-callback-thread done file5.ext
[info] application - play-akka.actor.default-dispatcher-5 finished file5.ext
答案 1 :(得分:0)
我认为可以在Java中实现迭代。
在这个问题中,有一个由Sadache在Scala中执行此操作的示例:Play 2.x : Reactive file upload with Iteratees
请注意,虽然没有可用于S3的异步api库,但如果您使用官方的amazon api java库,那么您将在上载的那一端阻塞。