Application.java 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. package controllers;
  2. import akka.dispatch.ExecutionContexts;
  3. import com.fasterxml.jackson.databind.ObjectMapper;
  4. import com.fasterxml.jackson.databind.node.ObjectNode;
  5. import models.World;
  6. import play.Play;
  7. import play.core.NamedThreadFactory;
  8. import play.libs.F;
  9. import play.libs.Json;
  10. import play.mvc.Controller;
  11. import play.mvc.Result;
  12. import scala.concurrent.ExecutionContext;
  13. import utils.Predicate;
  14. import utils.Predicated;
  15. import java.util.ArrayList;
  16. import java.util.List;
  17. import java.util.Random;
  18. import java.util.concurrent.LinkedBlockingQueue;
  19. import java.util.concurrent.ThreadLocalRandom;
  20. import java.util.concurrent.ThreadPoolExecutor;
  21. import java.util.concurrent.TimeUnit;
  22. public class Application extends Controller {
  23. private static final int MAX_QUERIES_PER_REQUEST = 20;
  24. private static final int TEST_DATABASE_ROWS = 10000;
  25. private static final ObjectMapper OBJECT_MAPPER = new ObjectMapper();
  26. private static final int partitionCount = Play.application().configuration().getInt("db.default.partitionCount");
  27. private static final int maxConnections =
  28. partitionCount * Play.application().configuration().getInt("db.default.maxConnectionsPerPartition");
  29. private static final int minConnections =
  30. partitionCount * Play.application().configuration().getInt("db.default.minConnectionsPerPartition");
  31. private static final ThreadPoolExecutor tpe = new ThreadPoolExecutor(minConnections, maxConnections,
  32. 0L, TimeUnit.MILLISECONDS,
  33. new LinkedBlockingQueue<Runnable>(),
  34. new NamedThreadFactory("dbEc"));
  35. private static final ExecutionContext dbEc = ExecutionContexts.fromExecutorService(tpe);
  36. public static Result json() {
  37. final ObjectNode result = OBJECT_MAPPER.createObjectNode();
  38. result.put("message", "Hello World!");
  39. return ok(result);
  40. }
  41. // If the thread-pool used by the database grows too large then our server
  42. // is probably struggling, and we should start dropping requests. Set
  43. // the max size of our queue something above the number of concurrent
  44. // connections that we need to handle.
  45. public static class IsDbAvailable implements Predicate {
  46. @Override
  47. public boolean condition() {
  48. return tpe.getQueue().size() <= 1024;
  49. }
  50. }
  51. @Predicated(predicate = IsDbAvailable.class, failed = SERVICE_UNAVAILABLE)
  52. public static F.Promise<Result> db() {
  53. return getRandomWorlds(1).map(new F.Function<List<World>, Result>() {
  54. @Override
  55. public Result apply(List<World> worlds) {
  56. return ok(Json.toJson(worlds.get(0)));
  57. }
  58. });
  59. }
  60. @Predicated(predicate = IsDbAvailable.class, failed = SERVICE_UNAVAILABLE)
  61. public static F.Promise<Result> queries(final String queryCountString) {
  62. int queryCount;
  63. try {
  64. queryCount = Integer.parseInt(queryCountString, 10);
  65. } catch (NumberFormatException e) {
  66. queryCount = 1;
  67. }
  68. if (queryCount < 1) {
  69. queryCount = 1;
  70. } else if (queryCount > 500) {
  71. queryCount = 500;
  72. }
  73. return getRandomWorlds(queryCount).map(new F.Function<List<World>, Result>() {
  74. @Override
  75. public Result apply(List<World> worlds) {
  76. return ok(Json.toJson(worlds));
  77. }
  78. });
  79. }
  80. private static F.Promise<List<World>> getRandomWorlds(final int n) {
  81. return F.Promise.promise(new F.Function0<List<World>>() {
  82. @Override
  83. public List<World> apply() throws Throwable {
  84. Random random = ThreadLocalRandom.current();
  85. List<World> worlds = new ArrayList<World>(n);
  86. for (int i = 0; i < n; ++i) {
  87. long randomId = random.nextInt(TEST_DATABASE_ROWS) + 1;
  88. World world = World.findById(randomId);
  89. worlds.add(world);
  90. }
  91. return worlds;
  92. }
  93. }, dbEc);
  94. }
  95. }