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

HubSignatureDirectives.scala « server « rorschach « getbootstrap « com « scala « main « src - github.com/twbs/rorschach.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: 9de2f15dcd24dbf2a9ce55c7af4c8057bfc0f1d5 (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
47
package com.chrisrebert.lmvtfy.server

import scala.util.{Try,Success,Failure}
import spray.routing.{Directive1, MalformedHeaderRejection, MalformedRequestContentRejection, ValidationRejection}
import spray.routing.directives.{BasicDirectives, HeaderDirectives, RouteDirectives, MarshallingDirectives}
import com.chrisrebert.lmvtfy.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