-
Notifications
You must be signed in to change notification settings - Fork 96
/
Copy pathHomeworkSpec.scala
128 lines (118 loc) · 4.46 KB
/
HomeworkSpec.scala
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
package com.evolutiongaming.bootcamp.json
import java.time.format.DateTimeFormatter
import java.time.{LocalDate, ZonedDateTime}
import cats.instances.either._
import cats.instances.list._
import cats.syntax.traverse._
import io.circe
import io.circe.parser._
import io.circe.generic.JsonCodec
import org.scalatest.EitherValues
import org.scalatest.matchers.must.Matchers
import org.scalatest.wordspec.AnyWordSpec
import scalaj.http.Http
import scala.io.Source
import scala.util.Try
/** HOMEWORK:
*
* Some classes and generated JSON codecs are provided for NBA API.
* Unfortunately, they don't work as expected out of the box.
* The task is to fix (rewrite) some of the codecs to make tests pass.
* You are not supposed to change anything in _class_ HomeworkSpec,
* instead of it you are supposed to keep your changes inside _companion object_ for HomeworkSpec.
*
* You are not allowed to rename fields in case classes.
* You are not allowed to remove fields from case classes.
* You are supposed to use camelCase if you introduce additional fields in case classes.
*
* It would be nice to avoid using Encoder/Decoder.forProductN where you specify all field names
*/
class HomeworkSpec extends AnyWordSpec with Matchers with EitherValues {
import HomeworkSpec._
"NBA JSON API client" should {
"get info about today games" in {
val date = LocalDate.now()
val scoreboardOrError = fetchScoreboard(date)
val scoreboard = scoreboardOrError.getOrElse(fail(scoreboardOrError.toString))
val allGameIds = scoreboard.games.map(_.gameId)
val gameInfosOrError = allGameIds.map(fetchGameInfo(date, _)).sequence
gameInfosOrError.getOrElse(fail(gameInfosOrError.toString))
succeed
}
"fetch games for 14 Feb 2020" in {
val date = LocalDate.of(2020, 2, 14)
val scoreboardOrError = fetchScoreboard(date)
val scoreboard = scoreboardOrError.getOrElse(fail(scoreboardOrError.toString))
val allGameIds = scoreboard.games.map(_.gameId)
val gameInfosOrError = allGameIds.map(fetchGameInfo(date, _)).sequence
val gameInfos = gameInfosOrError.getOrElse(fail(gameInfosOrError.toString))
gameInfos.size must be(1)
}
}
}
object HomeworkSpec {
@JsonCodec final case class TeamTotals(assists: String, fullTimeoutRemaining: String, plusMinus: String)
@JsonCodec final case class TeamBoxScore(totals: TeamTotals)
@JsonCodec final case class GameStats(hTeam: TeamBoxScore, vTeam: TeamBoxScore)
@JsonCodec final case class PrevMatchup(gameDate: LocalDate, gameId: String)
@JsonCodec final case class BoxScore(
basicGameData: Game,
previousMatchup: PrevMatchup,
stats: Option[GameStats],
)
@JsonCodec final case class JustScore(score: String)
@JsonCodec final case class TeamStats(
linescore: List[JustScore],
loss: String,
score: String,
teamId: String,
triCode: String,
)
@JsonCodec final case class GameDuration(hours: String, minutes: String)
@JsonCodec final case class Arena(
city: String,
country: String,
isDomestic: Boolean,
name: String,
stateAbbr: String,
)
@JsonCodec final case class Game(
arena: Arena,
attendance: String,
endTimeUTC: Option[ZonedDateTime],
gameDuration: GameDuration,
gameId: String,
gameUrlCode: String,
hTeam: TeamStats,
isBuzzerBeater: Boolean,
startTimeUTC: ZonedDateTime,
vTeam: TeamStats,
)
@JsonCodec final case class Scoreboard(games: List[Game], numGames: Int)
private def fetchScoreboard(date: LocalDate): Either[circe.Error, Scoreboard] = {
val dateString = date.format(DateTimeFormatter.BASIC_ISO_DATE)
val jsonString = Try {
val src = Source.fromResource(s"scoreboard_$dateString.json")
val s = src.mkString
src.close()
s
}.getOrElse {
val url = s"https://data.nba.net/10s/prod/v1/$dateString/scoreboard.json"
Http(url).asString.body
}
decode[Scoreboard](jsonString)
}
private def fetchGameInfo(date: LocalDate, gameId: String): Either[circe.Error, BoxScore] = {
val jsonString = Try {
val src = Source.fromResource(s"${gameId}_boxscore.json")
val s = src.mkString
src.close()
s
}.getOrElse {
val dateString = date.format(DateTimeFormatter.BASIC_ISO_DATE)
val url = s"https://data.nba.net/10s/prod/v1/$dateString/${gameId}_boxscore.json"
Http(url).asString.body
}
decode[BoxScore](jsonString)
}
}