Application.java 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. package controllers;
  2. import static models.tables.Fortune.FORTUNE;
  3. import static models.tables.World.WORLD;
  4. import static play.mvc.Http.MimeTypes.JSON;
  5. import java.util.Collections;
  6. import java.util.List;
  7. import java.util.Random;
  8. import java.util.concurrent.CompletableFuture;
  9. import java.util.concurrent.CompletionStage;
  10. import java.util.concurrent.ThreadLocalRandom;
  11. import javax.inject.Inject;
  12. import org.jooq.DSLContext;
  13. import org.jooq.JSONFormat;
  14. import org.jooq.JSONFormat.RecordFormat;
  15. import org.jooq.SQLDialect;
  16. import org.jooq.impl.DSL;
  17. import org.jooq.types.UInteger;
  18. import models.tables.records.FortuneRecord;
  19. import models.tables.records.WorldRecord;
  20. import play.db.Database;
  21. import play.mvc.Controller;
  22. import play.mvc.Result;
  23. import utils.DatabaseExecutionContext;
  24. public class Application extends Controller {
  25. private static final SQLDialect DIALECT = SQLDialect.MYSQL_5_7;
  26. private static final JSONFormat JSON_FORMAT = JSONFormat.DEFAULT_FOR_RECORDS.recordFormat(RecordFormat.OBJECT);
  27. private final Database db;
  28. private final DatabaseExecutionContext dbEc;
  29. @Inject
  30. public Application(final Database db, final DatabaseExecutionContext dbEc) {
  31. this.db = db;
  32. this.dbEc = dbEc;
  33. }
  34. public CompletionStage<Result> db() {
  35. return getRandomWorlds(1).thenApply(worlds -> ok(worlds.get(0).formatJSON(JSON_FORMAT)).as(JSON));
  36. }
  37. public CompletionStage<Result> queries(final String queries) {
  38. return getRandomWorlds(queryCount(queries)).thenApply(worlds -> ok(worlds.formatJSON(JSON_FORMAT)).as(JSON));
  39. }
  40. public CompletionStage<Result> fortunes() {
  41. return CompletableFuture.supplyAsync(() -> {
  42. final List<FortuneRecord> fortunes = this.db.withConnection(connection -> {
  43. return DSL.using(connection, DIALECT).select(FORTUNE.ID, FORTUNE.MESSAGE).from(FORTUNE).fetchInto(FortuneRecord.class);
  44. });
  45. fortunes.add(new FortuneRecord(UInteger.valueOf(0), "Additional fortune added at request time."));
  46. Collections.sort(fortunes, (f1, f2) -> f1.getMessage().compareTo(f2.getMessage()));
  47. return ok(views.html.fortunes.render(fortunes));
  48. }, dbEc);
  49. }
  50. public CompletionStage<Result> update(final String queries) {
  51. return getRandomWorlds(queryCount(queries)).thenApplyAsync(worlds -> {
  52. final Random random = ThreadLocalRandom.current();
  53. for (final WorldRecord world : worlds) {
  54. world.setRandomnumber((random.nextInt(10000) + 1));
  55. }
  56. final int batchSize = 25;
  57. final int batches = ((worlds.size() / batchSize) + 1);
  58. this.db.withConnection(connection -> {
  59. final DSLContext sql = DSL.using(connection, DIALECT);
  60. for ( int i = 0 ; i < batches ; ++i ) {
  61. sql.batchUpdate(worlds.subList(i * batchSize, Math.min((i + 1) * batchSize, worlds.size()))).execute();
  62. }
  63. return null;
  64. });
  65. return ok(worlds.formatJSON(JSON_FORMAT)).as(JSON);
  66. }, dbEc);
  67. }
  68. private int queryCount(final String queryCountString) {
  69. int queryCount;
  70. try {
  71. queryCount = Integer.parseInt(queryCountString, 10);
  72. } catch (final NumberFormatException e) {
  73. queryCount = 1;
  74. }
  75. if (queryCount < 1) {
  76. queryCount = 1;
  77. } else if (queryCount > 500) {
  78. queryCount = 500;
  79. }
  80. return queryCount;
  81. }
  82. private CompletionStage<org.jooq.Result<WorldRecord>> getRandomWorlds(final int n) {
  83. return CompletableFuture.supplyAsync(() -> {
  84. final Random random = ThreadLocalRandom.current();
  85. org.jooq.Result<WorldRecord> worlds = null;
  86. for (int i = 0; i < n; ++i) {
  87. long randomId = random.nextInt(10000) + 1;
  88. final org.jooq.Result<WorldRecord> world = this.db.withConnection(connection -> {
  89. return DSL.using(connection, DIALECT).selectFrom(WORLD).where(WORLD.ID.eq(UInteger.valueOf(randomId))).fetch();
  90. });
  91. if(worlds == null) {
  92. worlds = world;
  93. } else {
  94. worlds.add(world.get(0));
  95. }
  96. }
  97. return worlds;
  98. }, dbEc);
  99. }
  100. }