Application.java 4.3 KB

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