package com.xsn.explorer.services import com.alexitc.playsonify.core.FutureOr.Implicits.{FutureOps, OrOps} import com.alexitc.playsonify.core.{FutureApplicationResult, FuturePaginatedResult} import com.alexitc.playsonify.models.ordering.{FieldOrdering, OrderingCondition, OrderingQuery} import com.alexitc.playsonify.models.pagination.{Count, PaginatedQuery, PaginatedResult} import com.alexitc.playsonify.validators.PaginatedQueryValidator import com.xsn.explorer.errors.IPAddressFormatError import com.xsn.explorer.models.IPAddress import com.xsn.explorer.models.fields.MasternodeField import com.xsn.explorer.models.rpc.Masternode import com.xsn.explorer.parsers.MasternodeOrderingParser import javax.inject.Inject import org.scalactic.{Bad, Good} import scala.concurrent.ExecutionContext class MasternodeService @Inject() ( queryValidator: PaginatedQueryValidator, masternodeOrderingParser: MasternodeOrderingParser, xsnService: XSNService)( implicit ec: ExecutionContext) { def getMasternodes(paginatedQuery: PaginatedQuery, orderingQuery: OrderingQuery): FuturePaginatedResult[Masternode] = { val result = for { validatedQuery <- queryValidator.validate(paginatedQuery, 2000).toFutureOr ordering <- masternodeOrderingParser.from(orderingQuery).toFutureOr masternodes <- xsnService.getMasternodes().toFutureOr } yield build(masternodes, validatedQuery, ordering) result.toFuture } def getMasternode(ipAddressString: String): FutureApplicationResult[Masternode] = { val result = for { ipAddress <- IPAddress .from(ipAddressString) .map(Good(_)) .getOrElse(Bad(IPAddressFormatError).accumulating) .toFutureOr masternode <- xsnService.getMasternode(ipAddress).toFutureOr } yield masternode result.toFuture } private def build(list: List[Masternode], query: PaginatedQuery, ordering: FieldOrdering[MasternodeField]) = { val partial = sort(list, ordering) .slice(query.offset.int, query.offset.int + query.limit.int) PaginatedResult(query.offset, query.limit, Count(list.size), partial) } private def sort(list: List[Masternode], ordering: FieldOrdering[MasternodeField]) = { val sorted = sortByField(list, ordering.field) applyOrderingCondition(sorted, ordering.orderingCondition) } private def sortByField(list: List[Masternode], field: MasternodeField) = field match { case MasternodeField.ActiveSeconds => list.sortBy(_.activeSeconds) case MasternodeField.IP => list.sortBy(_.ip) case MasternodeField.LastSeen => list.sortBy(_.lastSeen) case MasternodeField.Status => list.sortBy(_.status) } private def applyOrderingCondition[A](list: List[A], orderingCondition: OrderingCondition) = orderingCondition match { case OrderingCondition.AscendingOrder => list case OrderingCondition.DescendingOrder => list.reverse } }