From 191d34e55342cf2f1b8af0512f5358326693c780 Mon Sep 17 00:00:00 2001 From: vlad Date: Thu, 2 Feb 2017 19:01:00 -0500 Subject: Moved email and phone number to core --- src/main/scala/xyz/driver/core/auth.scala | 4 ++++ src/main/scala/xyz/driver/core/domain.scala | 34 +++++++++++++++++++++++++++++ src/main/scala/xyz/driver/core/json.scala | 26 +++++++++++++++++++--- 3 files changed, 61 insertions(+), 3 deletions(-) create mode 100644 src/main/scala/xyz/driver/core/domain.scala (limited to 'src/main') diff --git a/src/main/scala/xyz/driver/core/auth.scala b/src/main/scala/xyz/driver/core/auth.scala index a9f52e5..156931e 100644 --- a/src/main/scala/xyz/driver/core/auth.scala +++ b/src/main/scala/xyz/driver/core/auth.scala @@ -1,5 +1,7 @@ package xyz.driver.core +import xyz.driver.core.domain.Email + object auth { trait Permission @@ -17,4 +19,6 @@ object auth { final case class RefreshToken(value: String) final case class PasswordHash(value: String) + + final case class AuthCredentials(email: Email, password: String) } diff --git a/src/main/scala/xyz/driver/core/domain.scala b/src/main/scala/xyz/driver/core/domain.scala new file mode 100644 index 0000000..f2629ee --- /dev/null +++ b/src/main/scala/xyz/driver/core/domain.scala @@ -0,0 +1,34 @@ +package xyz.driver.core + +object domain { + + final case class Email(username: String, domain: String) { + override def toString = username + "@" + domain + } + + object Email { + def parse(emailString: String): Option[Email] = { + Some(emailString.split("@")) collect { + case Array(username, domain) => Email(username, domain) + } + } + } + + final case class PhoneNumber(countryCode: String = "1", number: String) { + override def toString: String = s"+$countryCode $number" + } + + object PhoneNumber { + def parse(phoneNumberString: String): Option[PhoneNumber] = { + val onlyDigits = phoneNumberString.replaceAll("[^\\d.]", "") + + if (onlyDigits.length < 10) None + else { + val tenDigitNumber = onlyDigits.takeRight(10) + val countryCode = Option(onlyDigits.dropRight(10)).filter(_.nonEmpty).getOrElse("1") + + Some(PhoneNumber(countryCode, tenDigitNumber)) + } + } + } +} diff --git a/src/main/scala/xyz/driver/core/json.scala b/src/main/scala/xyz/driver/core/json.scala index 664ef48..3c0d8d4 100644 --- a/src/main/scala/xyz/driver/core/json.scala +++ b/src/main/scala/xyz/driver/core/json.scala @@ -5,13 +5,16 @@ import akka.http.scaladsl.server.PathMatcher.{Matched, Unmatched} import akka.http.scaladsl.server.{PathMatcher, _} import akka.http.scaladsl.unmarshalling.Unmarshaller import spray.json.{DeserializationException, JsNumber, _} +import xyz.driver.core.auth.AuthCredentials import xyz.driver.core.revision.Revision import xyz.driver.core.time.Time import xyz.driver.core.date.Date +import xyz.driver.core.domain.{Email, PhoneNumber} import scala.reflect.runtime.universe._ object json { + import DefaultJsonProtocol._ def IdInPath[T]: PathMatcher1[Id[T]] = new PathMatcher1[Id[T]] { def apply(path: Path) = path match { @@ -74,9 +77,8 @@ object json { Date .fromString(dateString) .getOrElse( - throw new DeserializationException( - s"Misformated ISO 8601 Date. Expected YYYY-MM-DD, but got $dateString.")) - case _ => throw new DeserializationException(s"Date expects a string, but got $value.") + throw DeserializationException(s"Misformated ISO 8601 Date. Expected YYYY-MM-DD, but got $dateString.")) + case _ => throw DeserializationException(s"Date expects a string, but got $value.") } } @@ -106,6 +108,24 @@ object json { } } + implicit val emailFormat = new RootJsonFormat[Email] { + def write(email: Email) = JsString(email.username + "@" + email.domain) + def read(json: JsValue): Email = json match { + + case JsString(value) => + Email.parse(value).getOrElse { + deserializationError("Expected '@' symbol in email string as Email, but got " + json) + } + + case _ => + deserializationError("Expected string as Email, but got " + json) + } + } + + implicit val phoneNumberFormat = jsonFormat2(PhoneNumber.apply) + + implicit val authCredentialsFormat = jsonFormat2(AuthCredentials) + class EnumJsonFormat[T](mapping: (String, T)*) extends RootJsonFormat[T] { private val map = mapping.toMap -- cgit v1.2.3