/** * Generated by API Builder - https://www.apibuilder.io * Service version: 1.0.1 * User agent: apibuilder app.apibuilder.io/gilt/gilt-public-api/1.0.1/http4s_0_15 */ package com.gilt.public.api.models { /** * @param name Product name * @param product URL to product object * @param id Unique identifier for product * @param brand Brand name * @param url Link to product detail page where item can be purchased * @param categories A list of categories to which this product belongs. * @param imageUrls A map of key/value pairs with the key being the image resolution WIDTHxHEIGHT * and the value being an array of image objects. * @param skus List of Skus */ final case class Product( name: String, product: String, id: Long, brand: String, url: String, content: com.gilt.public.api.models.ProductContent, categories: Seq[String], imageUrls: Map[String, _root_.io.circe.Json], skus: Seq[com.gilt.public.api.models.Sku] ) /** * @param categories A list of categories to which this product belongs. */ final case class ProductCategories( categories: Seq[String] ) /** * @param description Product description. * @param fitNotes Sizing information. * @param material Materials list. * @param careInstructions Additional care information. * @param origin Place of manufacture. */ final case class ProductContent( description: _root_.scala.Option[String] = None, fitNotes: _root_.scala.Option[String] = None, material: _root_.scala.Option[String] = None, careInstructions: _root_.scala.Option[String] = None, origin: _root_.scala.Option[String] = None ) /** * @param name Sale name * @param sale URL to single sale object * @param saleKey unique identifier for sale * @param store Store key * @param description A description of the sale's theme or brand * @param saleUrl Permalink to sale website * @param begins ISO8601-formatted time for beginning of sale * @param ends ISO-8601-formatted time for end of sale * @param imageUrls A map of key/value pairs with the key being the image resolution WIDTHxHEIGHT * and the value being an array of image objects. * @param products List of URLs to individual product objects (not available if sale is upcoming or * sold out) */ final case class SaleDetail( name: String, sale: String, saleKey: String, store: com.gilt.public.api.models.Store, description: _root_.scala.Option[String] = None, saleUrl: String, begins: _root_.java.time.Instant, ends: _root_.scala.Option[_root_.java.time.Instant] = None, imageUrls: Map[String, _root_.io.circe.Json], products: _root_.scala.Option[Seq[String]] = None ) /** * @param sales List of sales. */ final case class SaleList( sales: Seq[com.gilt.public.api.models.SaleDetail] ) /** * @param id The SKU unique identifier * @param inventoryStatus Describes product availability. One of: sold out, for sale or reserved. * @param msrpPrice MSRP price of SKU in US Dollars. * @param salePrice Sale price of SKU in US Dollars. * @param shippingSurcharge If absent, standard Gilt.com shipping policy and any resulting charges apply. If * present, standard shipping charge is overridden by amount listed here in US * Dollars. * @param attributes SKU attributes like color and size. */ final case class Sku( id: Long, inventoryStatus: String, msrpPrice: BigDecimal, salePrice: BigDecimal, shippingSurcharge: _root_.scala.Option[BigDecimal] = None, attributes: Seq[Map[String, _root_.io.circe.Json]] ) /** * Gilt.com is divided into stores. Currently there are four stores exposed by the * API: Women, Men, Baby & Kids and Home. */ sealed trait Store extends _root_.scala.Product with _root_.scala.Serializable object Store { case object Women extends Store { override def toString = "women" } case object Men extends Store { override def toString = "men" } case object Kids extends Store { override def toString = "kids" } case object Home extends Store { override def toString = "home" } /** * UNDEFINED captures values that are sent either in error or * that were added by the server after this library was * generated. We want to make it easy and obvious for users of * this library to handle this case gracefully. * * We use all CAPS for the variable name to avoid collisions * with the camel cased values above. */ final case class UNDEFINED(override val toString: String) extends Store /** * all returns a list of all the valid, known values. We use * lower case to avoid collisions with the camel cased values * above. */ val all: scala.List[Store] = scala.List(Women, Men, Kids, Home) private val byName: Map[String, Store] = all.map(x => x.toString.toLowerCase -> x).toMap def apply(value: String): Store = fromString(value).getOrElse(UNDEFINED(value)) def fromString(value: String): _root_.scala.Option[Store] = byName.get(value.toLowerCase) } } package com.gilt.public.api.models { package object json { import io.circe.Decoder._ import io.circe.Encoder._ import scala.language.implicitConversions // See below - Make Scala 2.11 Either monadic import scala.util.Try import io.circe.{Json, JsonObject, Encoder, Decoder, DecodingFailure} import io.circe.syntax._ import com.gilt.public.api.models.json._ // Make Scala 2.11 Either monadic private[api] implicit def eitherOps[A,B](e: Either[A,B]) = cats.implicits.catsSyntaxEither(e) private[api] implicit val decodeUUID: Decoder[_root_.java.util.UUID] = Decoder.decodeString.emapTry(str => Try(_root_.java.util.UUID.fromString(str))) private[api] implicit val encodeUUID: Encoder[_root_.java.util.UUID] = Encoder.encodeString.contramap[_root_.java.util.UUID](uuid => uuid.toString) private[api] implicit val decodeInstant: Decoder[_root_.java.time.Instant] = Decoder.decodeString.emapTry(str => Try(_root_.java.time.OffsetDateTime.parse(str).toInstant)) private[api] implicit val encodeInstant: Encoder[_root_.java.time.Instant] = Encoder.encodeString.contramap[_root_.java.time.Instant](_.toString) private[api] implicit val decodeLocalDate: Decoder[_root_.java.time.LocalDate] = Decoder.decodeString.emapTry(str => Try(_root_.java.time.LocalDate.parse(str))) private[api] implicit val encodeLocalDate: Encoder[_root_.java.time.LocalDate] = Encoder.encodeString.contramap[_root_.java.time.LocalDate](_.toString) implicit val jsonDecoderGiltPublicApiStore: Decoder[com.gilt.public.api.models.Store] = Decoder.decodeString.map(com.gilt.public.api.models.Store(_)) implicit val jsonEncoderGiltPublicApiStore: Encoder[com.gilt.public.api.models.Store] = Encoder.encodeString.contramap[com.gilt.public.api.models.Store](_.toString) implicit def decodeGiltPublicApiProduct: Decoder[Product] = Decoder.instance { c => for { name <- c.downField("name").as[String] product <- c.downField("product").as[String] id <- c.downField("id").as[Long] brand <- c.downField("brand").as[String] url <- c.downField("url").as[String] content <- c.downField("content").as[com.gilt.public.api.models.ProductContent] categories <- c.downField("categories").as[Seq[String]] imageUrls <- c.downField("image_urls").as[Map[String, _root_.io.circe.Json]] skus <- c.downField("skus").as[Seq[com.gilt.public.api.models.Sku]] } yield { Product( name = name, product = product, id = id, brand = brand, url = url, content = content, categories = categories, imageUrls = imageUrls, skus = skus ) } } implicit def encodeGiltPublicApiProduct: Encoder[Product] = Encoder.instance { t => Json.fromFields(Seq( Some("name" -> t.name.asJson), Some("product" -> t.product.asJson), Some("id" -> t.id.asJson), Some("brand" -> t.brand.asJson), Some("url" -> t.url.asJson), Some("content" -> t.content.asJson), Some("categories" -> t.categories.asJson), Some("image_urls" -> t.imageUrls.asJson), Some("skus" -> t.skus.asJson) ).flatten) } implicit def decodeGiltPublicApiProductCategories: Decoder[ProductCategories] = Decoder.instance { c => for { categories <- c.downField("categories").as[Seq[String]] } yield { ProductCategories( categories = categories ) } } implicit def encodeGiltPublicApiProductCategories: Encoder[ProductCategories] = Encoder.instance { t => Json.fromFields(Seq( Some("categories" -> t.categories.asJson) ).flatten) } implicit def decodeGiltPublicApiProductContent: Decoder[ProductContent] = Decoder.instance { c => for { description <- c.downField("description").as[Option[String]] fitNotes <- c.downField("fit_notes").as[Option[String]] material <- c.downField("material").as[Option[String]] careInstructions <- c.downField("care_instructions").as[Option[String]] origin <- c.downField("origin").as[Option[String]] } yield { ProductContent( description = description, fitNotes = fitNotes, material = material, careInstructions = careInstructions, origin = origin ) } } implicit def encodeGiltPublicApiProductContent: Encoder[ProductContent] = Encoder.instance { t => Json.fromFields(Seq( t.description.map(t => "description" -> t.asJson), t.fitNotes.map(t => "fit_notes" -> t.asJson), t.material.map(t => "material" -> t.asJson), t.careInstructions.map(t => "care_instructions" -> t.asJson), t.origin.map(t => "origin" -> t.asJson) ).flatten) } implicit def decodeGiltPublicApiSaleDetail: Decoder[SaleDetail] = Decoder.instance { c => for { name <- c.downField("name").as[String] sale <- c.downField("sale").as[String] saleKey <- c.downField("sale_key").as[String] store <- c.downField("store").as[com.gilt.public.api.models.Store] description <- c.downField("description").as[Option[String]] saleUrl <- c.downField("sale_url").as[String] begins <- c.downField("begins").as[_root_.java.time.Instant] ends <- c.downField("ends").as[Option[_root_.java.time.Instant]] imageUrls <- c.downField("image_urls").as[Map[String, _root_.io.circe.Json]] products <- c.downField("products").as[Option[Seq[String]]] } yield { SaleDetail( name = name, sale = sale, saleKey = saleKey, store = store, description = description, saleUrl = saleUrl, begins = begins, ends = ends, imageUrls = imageUrls, products = products ) } } implicit def encodeGiltPublicApiSaleDetail: Encoder[SaleDetail] = Encoder.instance { t => Json.fromFields(Seq( Some("name" -> t.name.asJson), Some("sale" -> t.sale.asJson), Some("sale_key" -> t.saleKey.asJson), Some("store" -> t.store.asJson), t.description.map(t => "description" -> t.asJson), Some("sale_url" -> t.saleUrl.asJson), Some("begins" -> t.begins.asJson), t.ends.map(t => "ends" -> t.asJson), Some("image_urls" -> t.imageUrls.asJson), t.products.map(t => "products" -> t.asJson) ).flatten) } implicit def decodeGiltPublicApiSaleList: Decoder[SaleList] = Decoder.instance { c => for { sales <- c.downField("sales").as[Seq[com.gilt.public.api.models.SaleDetail]] } yield { SaleList( sales = sales ) } } implicit def encodeGiltPublicApiSaleList: Encoder[SaleList] = Encoder.instance { t => Json.fromFields(Seq( Some("sales" -> t.sales.asJson) ).flatten) } implicit def decodeGiltPublicApiSku: Decoder[Sku] = Decoder.instance { c => for { id <- c.downField("id").as[Long] inventoryStatus <- c.downField("inventory_status").as[String] msrpPrice <- c.downField("msrp_price").as[BigDecimal] salePrice <- c.downField("sale_price").as[BigDecimal] shippingSurcharge <- c.downField("shipping_surcharge").as[Option[BigDecimal]] attributes <- c.downField("attributes").as[Seq[Map[String, _root_.io.circe.Json]]] } yield { Sku( id = id, inventoryStatus = inventoryStatus, msrpPrice = msrpPrice, salePrice = salePrice, shippingSurcharge = shippingSurcharge, attributes = attributes ) } } implicit def encodeGiltPublicApiSku: Encoder[Sku] = Encoder.instance { t => Json.fromFields(Seq( Some("id" -> t.id.asJson), Some("inventory_status" -> t.inventoryStatus.asJson), Some("msrp_price" -> t.msrpPrice.asJson), Some("sale_price" -> t.salePrice.asJson), t.shippingSurcharge.map(t => "shipping_surcharge" -> t.asJson), Some("attributes" -> t.attributes.asJson) ).flatten) } } } package com.gilt.public.api { import org.http4s.client.blaze._ import io.circe.syntax._ object Constants { val BaseUrl = "https://api.gilt.com/v1" val Namespace = "com.gilt.public.api" val UserAgent = "apibuilder app.apibuilder.io/gilt/gilt-public-api/1.0.1/http4s_0_15" val Version = "1.0.1" val VersionMajor = 1 } class Client( val baseUrl: org.http4s.Uri = org.http4s.Uri.unsafeFromString("https://api.gilt.com/v1"), auth: scala.Option[com.gilt.public.api.Authorization] = None, defaultHeaders: Seq[(String, String)] = Nil, asyncHttpClient: org.http4s.client.Client = Client.defaultAsyncHttpClient ) extends interfaces.Client { import org.http4s.Response import com.gilt.public.api.models.json._ def closeAsyncHttpClient(): Unit = { asyncHttpClient.shutdownNow() } def products: Products = Products def saleDetails: SaleDetails = SaleDetails def saleList: SaleList = SaleList object Products extends Products { override def getDetailJsonByProductId( productId: String, apikey: String, requestHeaders: Seq[(String, String)] = Nil ): scalaz.concurrent.Task[com.gilt.public.api.models.Product] = { val urlPath = Seq("products", productId, "detail.json") val queryParameters = Seq( Some("apikey" -> apikey) ).flatten _executeRequest[Unit, com.gilt.public.api.models.Product]("GET", path = urlPath, queryParameters = queryParameters, requestHeaders = requestHeaders) { case r if r.status.code == 200 => _root_.com.gilt.public.api.Client.parseJson[com.gilt.public.api.models.Product]("com.gilt.public.api.models.Product", r) case r => scalaz.concurrent.Task.fail(new com.gilt.public.api.errors.FailedRequest(r.status.code, s"Unsupported response code[${r.status.code}]. Expected: 200")) } } override def getCategoriesJson( apikey: String, requestHeaders: Seq[(String, String)] = Nil ): scalaz.concurrent.Task[com.gilt.public.api.models.ProductCategories] = { val urlPath = Seq("products", "categories.json") val queryParameters = Seq( Some("apikey" -> apikey) ).flatten _executeRequest[Unit, com.gilt.public.api.models.ProductCategories]("GET", path = urlPath, queryParameters = queryParameters, requestHeaders = requestHeaders) { case r if r.status.code == 200 => _root_.com.gilt.public.api.Client.parseJson[com.gilt.public.api.models.ProductCategories]("com.gilt.public.api.models.ProductCategories", r) case r => scalaz.concurrent.Task.fail(new com.gilt.public.api.errors.FailedRequest(r.status.code, s"Unsupported response code[${r.status.code}]. Expected: 200")) } } } object SaleDetails extends SaleDetails { override def getDetailJson( store: com.gilt.public.api.models.Store, sale: String, apikey: String, requestHeaders: Seq[(String, String)] = Nil ): scalaz.concurrent.Task[com.gilt.public.api.models.SaleDetail] = { val urlPath = Seq("sales", store.toString, ":saledetail.json") val queryParameters = Seq( Some("apikey" -> apikey) ).flatten _executeRequest[Unit, com.gilt.public.api.models.SaleDetail]("GET", path = urlPath, queryParameters = queryParameters, requestHeaders = requestHeaders) { case r if r.status.code == 200 => _root_.com.gilt.public.api.Client.parseJson[com.gilt.public.api.models.SaleDetail]("com.gilt.public.api.models.SaleDetail", r) case r => scalaz.concurrent.Task.fail(new com.gilt.public.api.errors.FailedRequest(r.status.code, s"Unsupported response code[${r.status.code}]. Expected: 200")) } } } object SaleList extends SaleList { override def getActiveJson( apikey: String, requestHeaders: Seq[(String, String)] = Nil ): scalaz.concurrent.Task[com.gilt.public.api.models.SaleList] = { val urlPath = Seq("sales", "active.json") val queryParameters = Seq( Some("apikey" -> apikey) ).flatten _executeRequest[Unit, com.gilt.public.api.models.SaleList]("GET", path = urlPath, queryParameters = queryParameters, requestHeaders = requestHeaders) { case r if r.status.code == 200 => _root_.com.gilt.public.api.Client.parseJson[com.gilt.public.api.models.SaleList]("com.gilt.public.api.models.SaleList", r) case r => scalaz.concurrent.Task.fail(new com.gilt.public.api.errors.FailedRequest(r.status.code, s"Unsupported response code[${r.status.code}]. Expected: 200")) } } override def getActiveJsonByStore( store: String, apikey: String, requestHeaders: Seq[(String, String)] = Nil ): scalaz.concurrent.Task[com.gilt.public.api.models.SaleList] = { val urlPath = Seq("sales", store, "active.json") val queryParameters = Seq( Some("apikey" -> apikey) ).flatten _executeRequest[Unit, com.gilt.public.api.models.SaleList]("GET", path = urlPath, queryParameters = queryParameters, requestHeaders = requestHeaders) { case r if r.status.code == 200 => _root_.com.gilt.public.api.Client.parseJson[com.gilt.public.api.models.SaleList]("com.gilt.public.api.models.SaleList", r) case r => scalaz.concurrent.Task.fail(new com.gilt.public.api.errors.FailedRequest(r.status.code, s"Unsupported response code[${r.status.code}]. Expected: 200")) } } override def getUpcomingJson( apikey: String, requestHeaders: Seq[(String, String)] = Nil ): scalaz.concurrent.Task[com.gilt.public.api.models.SaleList] = { val urlPath = Seq("sales", "upcoming.json") val queryParameters = Seq( Some("apikey" -> apikey) ).flatten _executeRequest[Unit, com.gilt.public.api.models.SaleList]("GET", path = urlPath, queryParameters = queryParameters, requestHeaders = requestHeaders) { case r if r.status.code == 200 => _root_.com.gilt.public.api.Client.parseJson[com.gilt.public.api.models.SaleList]("com.gilt.public.api.models.SaleList", r) case r => scalaz.concurrent.Task.fail(new com.gilt.public.api.errors.FailedRequest(r.status.code, s"Unsupported response code[${r.status.code}]. Expected: 200")) } } override def getUpcomingJsonByStore( store: String, apikey: String, requestHeaders: Seq[(String, String)] = Nil ): scalaz.concurrent.Task[com.gilt.public.api.models.SaleList] = { val urlPath = Seq("sales", store, "upcoming.json") val queryParameters = Seq( Some("apikey" -> apikey) ).flatten _executeRequest[Unit, com.gilt.public.api.models.SaleList]("GET", path = urlPath, queryParameters = queryParameters, requestHeaders = requestHeaders) { case r if r.status.code == 200 => _root_.com.gilt.public.api.Client.parseJson[com.gilt.public.api.models.SaleList]("com.gilt.public.api.models.SaleList", r) case r => scalaz.concurrent.Task.fail(new com.gilt.public.api.errors.FailedRequest(r.status.code, s"Unsupported response code[${r.status.code}]. Expected: 200")) } } } private lazy val defaultApiHeaders = Seq( ("User-Agent", Constants.UserAgent), ("X-Apidoc-Version", Constants.Version), ("X-Apidoc-Version-Major", Constants.VersionMajor.toString) ) def apiHeaders: Seq[(String, String)] = defaultApiHeaders def modifyRequest(request: scalaz.concurrent.Task[org.http4s.Request]): scalaz.concurrent.Task[org.http4s.Request] = request implicit def circeJsonEncoder[A](implicit encoder: io.circe.Encoder[A]) = org.http4s.circe.jsonEncoderOf[A] def _executeRequest[T, U]( method: String, path: Seq[String], queryParameters: Seq[(String, String)] = Nil, requestHeaders: Seq[(String, String)] = Nil, body: Option[T] = None, formBody : Option[org.http4s.UrlForm] = None )(handler: org.http4s.Response => scalaz.concurrent.Task[U] )(implicit encoder: io.circe.Encoder[T]): scalaz.concurrent.Task[U] = { import org.http4s.QueryParamEncoder._ val m = org.http4s.Method.fromString(method) match { case scalaz.\/-(m) => m case scalaz.-\/(e) => sys.error(e.toString) } val headers = org.http4s.Headers(( apiHeaders ++ defaultHeaders ++ requestHeaders ).groupBy(_._1).map { case (k, l) => org.http4s.Header(k, l.last._2) }.toList) val queryMap = queryParameters.groupBy(_._1).map { case (k, v) => k -> v.map(_._2) } val uri = path.foldLeft(baseUrl){ case (uri, segment) => uri / segment }.setQueryParams(queryMap) val request = org.http4s.Request(method = m, uri = uri, headers = headers) val reqAndMaybeAuth = auth.fold(request) { case Authorization.Basic(username, passwordOpt) => { val userpass = s"$username:${passwordOpt.getOrElse("")}" val token = java.util.Base64.getEncoder.encodeToString(userpass.getBytes(java.nio.charset.StandardCharsets.ISO_8859_1)) request.putHeaders(org.http4s.Header("Authorization", s"Basic $token")) } case a => sys.error("Invalid authorization scheme[" + a.getClass + "]") } val reqAndMaybeAuthAndBody = if (formBody.nonEmpty) formBody.fold(scalaz.concurrent.Task.now(reqAndMaybeAuth))(reqAndMaybeAuth.withBody) else body.fold(scalaz.concurrent.Task.now(reqAndMaybeAuth))(reqAndMaybeAuth.withBody) asyncHttpClient.fetch(modifyRequest(reqAndMaybeAuthAndBody))(handler) } } object Client { implicit def circeJsonDecoder[A](implicit decoder: io.circe.Decoder[A]) = org.http4s.circe.jsonOf[A] private lazy val defaultAsyncHttpClient = PooledHttp1Client() def parseJson[T]( className: String, r: org.http4s.Response )(implicit decoder: io.circe.Decoder[T]): scalaz.concurrent.Task[T] = r.attemptAs[T].run.flatMap { case scalaz.\/-(value) => scalaz.concurrent.Task.now(value) case scalaz.-\/(error) => scalaz.concurrent.Task.fail(new com.gilt.public.api.errors.FailedRequest(r.status.code, s"Invalid json for class[" + className + "]", None, error)) } } sealed trait Authorization extends _root_.scala.Product with _root_.scala.Serializable object Authorization { final case class Basic(username: String, password: Option[String] = None) extends Authorization } package interfaces { trait Client { def baseUrl: org.http4s.Uri def products: com.gilt.public.api.Products def saleDetails: com.gilt.public.api.SaleDetails def saleList: com.gilt.public.api.SaleList } } trait Products { def getDetailJsonByProductId( productId: String, apikey: String, requestHeaders: Seq[(String, String)] = Nil ): scalaz.concurrent.Task[com.gilt.public.api.models.Product] def getCategoriesJson( apikey: String, requestHeaders: Seq[(String, String)] = Nil ): scalaz.concurrent.Task[com.gilt.public.api.models.ProductCategories] } trait SaleDetails { def getDetailJson( store: com.gilt.public.api.models.Store, sale: String, apikey: String, requestHeaders: Seq[(String, String)] = Nil ): scalaz.concurrent.Task[com.gilt.public.api.models.SaleDetail] } trait SaleList { def getActiveJson( apikey: String, requestHeaders: Seq[(String, String)] = Nil ): scalaz.concurrent.Task[com.gilt.public.api.models.SaleList] def getActiveJsonByStore( store: String, apikey: String, requestHeaders: Seq[(String, String)] = Nil ): scalaz.concurrent.Task[com.gilt.public.api.models.SaleList] def getUpcomingJson( apikey: String, requestHeaders: Seq[(String, String)] = Nil ): scalaz.concurrent.Task[com.gilt.public.api.models.SaleList] def getUpcomingJsonByStore( store: String, apikey: String, requestHeaders: Seq[(String, String)] = Nil ): scalaz.concurrent.Task[com.gilt.public.api.models.SaleList] } package errors { final case class FailedRequest(responseCode: Int, message: String, requestUri: Option[_root_.java.net.URI] = None, parent: Exception = null) extends _root_.java.lang.Exception(s"HTTP $responseCode: $message", parent) } }