224 lines
7.5 KiB
Scala
224 lines
7.5 KiB
Scala
package controllers
|
|
|
|
import play.api.i18n.Messages.Implicits._
|
|
import play.api.libs.json._
|
|
import play.api.mvc._
|
|
import play.api.Play.current
|
|
|
|
import lila.api.Context
|
|
import lila.app._
|
|
import lila.puzzle.{ PuzzleId, Result, Puzzle => PuzzleModel, UserInfos }
|
|
import lila.game.PgnDump
|
|
import lila.user.UserRepo
|
|
import views._
|
|
|
|
object Puzzle extends LilaController {
|
|
|
|
private def env = Env.puzzle
|
|
|
|
private def renderJson(
|
|
puzzle: PuzzleModel,
|
|
userInfos: Option[UserInfos],
|
|
mode: String,
|
|
voted: Option[Boolean],
|
|
round: Option[lila.puzzle.Round] = None,
|
|
result: Option[Result] = None
|
|
)(implicit ctx: Context): Fu[JsObject] = env.jsonView(
|
|
puzzle = puzzle,
|
|
userInfos = userInfos,
|
|
round = round,
|
|
mode = mode,
|
|
isMobileApi = ctx.isMobileApi,
|
|
result = result,
|
|
voted = voted
|
|
)
|
|
|
|
private def renderShow(puzzle: PuzzleModel, mode: String)(implicit ctx: Context) =
|
|
env userInfos ctx.me flatMap { infos =>
|
|
renderJson(puzzle = puzzle, userInfos = infos, mode = mode, voted = none) map { json =>
|
|
views.html.puzzle.show(puzzle, data = json, pref = env.jsonView.pref(ctx.pref))
|
|
}
|
|
}
|
|
|
|
def daily = Open { implicit ctx =>
|
|
OptionFuResult(env.daily.get flatMap {
|
|
_.map(_.id) ?? env.api.puzzle.find
|
|
}) { puzzle =>
|
|
negotiate(
|
|
html = renderShow(puzzle, "play") map { Ok(_) },
|
|
api = _ => puzzleJson(puzzle) map { Ok(_) }
|
|
) map { NoCache(_) }
|
|
}
|
|
}
|
|
|
|
def home = Open { implicit ctx =>
|
|
env.selector(ctx.me) flatMap { puzzle =>
|
|
renderShow(puzzle, ctx.isAuth.fold("play", "try")) map { Ok(_) }
|
|
}
|
|
}
|
|
|
|
def show(id: PuzzleId) = Open { implicit ctx =>
|
|
OptionFuOk(env.api.puzzle find id) { puzzle =>
|
|
renderShow(puzzle, "play")
|
|
}
|
|
}
|
|
|
|
def load(id: PuzzleId) = Open { implicit ctx =>
|
|
XhrOnly {
|
|
OptionFuOk(env.api.puzzle find id)(puzzleJson) map (_ as JSON)
|
|
}
|
|
}
|
|
|
|
private def puzzleJson(puzzle: PuzzleModel)(implicit ctx: Context) =
|
|
(env userInfos ctx.me) flatMap { infos =>
|
|
renderJson(puzzle, infos, ctx.isAuth.fold("play", "try"), voted = none)
|
|
}
|
|
|
|
// XHR load next play puzzle
|
|
def newPuzzle = Open { implicit ctx =>
|
|
XhrOnly {
|
|
env.selector(ctx.me) zip (env userInfos ctx.me) flatMap {
|
|
case (puzzle, infos) =>
|
|
renderJson(puzzle, infos, ctx.isAuth.fold("play", "try"), voted = none) map { json =>
|
|
Ok(json) as JSON
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// mobile app BC
|
|
def round(id: PuzzleId) = OpenBody { implicit ctx =>
|
|
implicit val req = ctx.body
|
|
OptionFuResult(env.api.puzzle find id) { puzzle =>
|
|
if (puzzle.mate) lila.mon.puzzle.round.mate()
|
|
else lila.mon.puzzle.round.material()
|
|
env.forms.round.bindFromRequest.fold(
|
|
err => fuccess(BadRequest(errorsAsJson(err))),
|
|
resultInt => {
|
|
val result = Result(resultInt == 1)
|
|
ctx.me match {
|
|
case Some(me) => for {
|
|
(round, mode) <- env.finisher(puzzle, me, result)
|
|
me2 <- mode.rated.fold(UserRepo byId me.id map (_ | me), fuccess(me))
|
|
infos <- env userInfos me2
|
|
voted <- ctx.me.?? { env.api.vote.value(puzzle.id, _) }
|
|
data <- renderJson(puzzle, infos.some, "view", voted = voted, result = result.some, round = round.some)
|
|
} yield {
|
|
lila.mon.puzzle.round.user()
|
|
val d2 = if (mode.rated) data else data ++ Json.obj("win" -> result.win)
|
|
Ok(d2)
|
|
}
|
|
case None =>
|
|
lila.mon.puzzle.round.anon()
|
|
env.finisher.incPuzzleAttempts(puzzle)
|
|
renderJson(puzzle, none, "view", result = result.some, voted = none) map { data =>
|
|
val d2 = data ++ Json.obj("win" -> result.win)
|
|
Ok(d2)
|
|
}
|
|
}
|
|
}
|
|
) map (_ as JSON)
|
|
}
|
|
}
|
|
|
|
// new API
|
|
def round2(id: PuzzleId) = OpenBody { implicit ctx =>
|
|
implicit val req = ctx.body
|
|
OptionFuResult(env.api.puzzle find id) { puzzle =>
|
|
if (puzzle.mate) lila.mon.puzzle.round.mate()
|
|
else lila.mon.puzzle.round.material()
|
|
env.forms.round.bindFromRequest.fold(
|
|
err => fuccess(BadRequest(errorsAsJson(err))),
|
|
resultInt => ctx.me match {
|
|
case Some(me) => for {
|
|
(round, mode) <- env.finisher(puzzle, me, Result(resultInt == 1))
|
|
me2 <- mode.rated.fold(UserRepo byId me.id map (_ | me), fuccess(me))
|
|
infos <- env userInfos me2
|
|
voted <- ctx.me.?? { env.api.vote.value(puzzle.id, _) }
|
|
} yield {
|
|
lila.mon.puzzle.round.user()
|
|
Ok(Json.obj(
|
|
"user" -> lila.puzzle.JsonView.infos(false)(infos),
|
|
"round" -> lila.puzzle.JsonView.round(round),
|
|
"voted" -> voted
|
|
))
|
|
}
|
|
case None =>
|
|
lila.mon.puzzle.round.anon()
|
|
env.finisher.incPuzzleAttempts(puzzle)
|
|
Ok(Json.obj("user" -> false)).fuccess
|
|
}
|
|
) map (_ as JSON)
|
|
}
|
|
}
|
|
|
|
def vote(id: PuzzleId) = AuthBody { implicit ctx => me =>
|
|
implicit val req = ctx.body
|
|
env.forms.vote.bindFromRequest.fold(
|
|
err => fuccess(BadRequest(errorsAsJson(err))),
|
|
vote => env.api.vote.find(id, me) flatMap {
|
|
v => env.api.vote.update(id, me, v, vote == 1)
|
|
} map {
|
|
case (p, a) =>
|
|
if (vote == 1) lila.mon.puzzle.vote.up()
|
|
else lila.mon.puzzle.vote.down()
|
|
Ok(Json.arr(a.value, p.vote.sum))
|
|
}
|
|
) map (_ as JSON)
|
|
}
|
|
|
|
def recentGame = Action.async { req =>
|
|
if (!get("token", req).contains(Env.api.apiToken)) BadRequest.fuccess
|
|
else {
|
|
import akka.pattern.ask
|
|
import makeTimeout.short
|
|
Env.game.recentGoodGameActor ? true mapTo manifest[Option[String]] flatMap {
|
|
_ ?? lila.game.GameRepo.gameWithInitialFen flatMap {
|
|
case Some((game, initialFen)) =>
|
|
Env.api.pgnDump(game, initialFen.map(_.value), PgnDump.WithFlags(clocks = false)) map { pgn =>
|
|
Ok(pgn.render)
|
|
}
|
|
case _ =>
|
|
lila.log("puzzle import").info("No recent good game, serving a random one :-/")
|
|
lila.game.GameRepo.findRandomFinished(1000) flatMap {
|
|
_ ?? { game =>
|
|
lila.game.GameRepo.initialFen(game) flatMap { fen =>
|
|
Env.api.pgnDump(game, fen, PgnDump.WithFlags(clocks = false)) map { pgn =>
|
|
Ok(pgn.render)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
def importOne = Action.async(parse.json) { implicit req =>
|
|
env.api.puzzle.importOne(req.body, ~get("token", req)) map { id =>
|
|
val url = s"https://lichess.org/training/$id"
|
|
lila.log("puzzle import").info(s"${req.remoteAddress} $url")
|
|
Ok(s"kthxbye $url")
|
|
} recover {
|
|
case e =>
|
|
lila.log("puzzle import").warn(s"${req.remoteAddress} ${e.getMessage}", e)
|
|
BadRequest(e.getMessage)
|
|
}
|
|
}
|
|
|
|
def embed = Action { req =>
|
|
Ok {
|
|
val bg = get("bg", req) | "light"
|
|
val theme = get("theme", req) | "brown"
|
|
val url = s"""${req.domain + routes.Puzzle.frame}?bg=$bg&theme=$theme"""
|
|
s"""document.write("<iframe src='//$url&embed=" + document.domain + "' class='lichess-training-iframe' allowtransparency='true' frameBorder='0' style='width: 224px; height: 264px;' title='Lichess free online chess'></iframe>");"""
|
|
} as JAVASCRIPT withHeaders (CACHE_CONTROL -> "max-age=86400")
|
|
}
|
|
|
|
def frame = Open { implicit ctx =>
|
|
OptionOk(env.daily.get) { daily =>
|
|
html.puzzle.embed(daily)
|
|
}
|
|
}
|
|
}
|