Welcome to mirror list, hosted at ThFree Co, Russian Federation.

HubSignatureDirectives.scala « server « savage « getbootstrap « com « scala « main « src - github.com/twbs/savage.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: a0d4153459ae43413cc9b1aed1cbee7e023add99 (plain)
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
package com.getbootstrap.savage.server

import scala.util.{Try,Success,Failure}
import spray.routing.{Directive1, MalformedHeaderRejection, MalformedRequestContentRejection, ValidationRejection}
import spray.routing.directives.{BasicDirectives, HeaderDirectives, RouteDirectives, MarshallingDirectives}
import com.getbootstrap.savage.util.{HmacSha1,Utf8ByteArray}

trait HubSignatureDirectives  {
  import BasicDirectives.provide
  import HeaderDirectives.headerValueByName
  import RouteDirectives.reject
  import MarshallingDirectives.{entity, as}

  private val xHubSignature = "X-Hub-Signature"
  private val hubSignatureHeaderValue = headerValueByName(xHubSignature)

  val hubSignature: Directive1[Array[Byte]] = hubSignatureHeaderValue.flatMap { algoEqHex =>
    val bytesFromHexOption = algoEqHex.split('=') match {
      case Array("sha1", hex) => Try{ javax.xml.bind.DatatypeConverter.parseHexBinary(hex) }.toOption
      case _ => None
    }
    bytesFromHexOption match {
      case Some(bytesFromHex) => provide(bytesFromHex)
      case None => reject(MalformedHeaderRejection(xHubSignature, "Malformed HMAC"))
    }
  }

  private val bytesEntity = entity(as[Array[Byte]])

  def stringEntityMatchingHubSignature(secretKey: Array[Byte]): Directive1[String] = hubSignature.flatMap { signature =>
    bytesEntity.flatMap { dataBytes =>
      val hmac = new HmacSha1(mac = signature, secretKey = secretKey, data = dataBytes)
      if (hmac.isValid) {
        dataBytes.utf8String match {
          case Success(string) => provide(string)
          case Failure(exc) => reject(MalformedRequestContentRejection("Request body is not valid UTF-8", Some(exc)))
        }
      }
      else {
        reject(ValidationRejection("Incorrect HMAC"))
      }
    }
  }
}

object HubSignatureDirectives extends HubSignatureDirectives