aboutsummaryrefslogtreecommitdiff
path: root/kamon-core/src/main/scala/kamon/trace/IdentityProvider.scala
blob: 937200f5dd631cea29587b01f705f930140987c1 (plain) (blame)
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
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
package kamon.trace

import java.nio.ByteBuffer
import java.util.concurrent.ThreadLocalRandom

import kamon.util.HexCodec

import scala.util.Try

trait IdentityProvider {
  def traceIdGenerator(): IdentityProvider.Generator
  def spanIdGenerator(): IdentityProvider.Generator
}

object IdentityProvider {
  case class Identifier(string: String, bytes: Array[Byte]) {

    override def equals(obj: Any): Boolean = {
      if(obj != null && obj.isInstanceOf[Identifier])
        obj.asInstanceOf[Identifier].string == string
      else false
    }
  }

  val NoIdentifier = Identifier("", new Array[Byte](0))

  trait Generator {
    def generate(): Identifier
    def from(string: String): Identifier
    def from(bytes: Array[Byte]): Identifier
  }


  class Default extends IdentityProvider {
    protected val longGenerator = new Generator {
      override def generate(): Identifier = {
        val data = ByteBuffer.wrap(new Array[Byte](8))
        val random = ThreadLocalRandom.current().nextLong()
        data.putLong(random)

        Identifier(HexCodec.toLowerHex(random), data.array())
      }

      override def from(string: String): Identifier =  Try {
        val identifierLong = HexCodec.lowerHexToUnsignedLong(string)
        val data = ByteBuffer.allocate(8)
        data.putLong(identifierLong)

        Identifier(string, data.array())
      } getOrElse(IdentityProvider.NoIdentifier)

      override def from(bytes: Array[Byte]): Identifier = Try {
        val buffer = ByteBuffer.wrap(bytes)
        val identifierLong = buffer.getLong

        Identifier(HexCodec.toLowerHex(identifierLong), bytes)
      } getOrElse(IdentityProvider.NoIdentifier)
    }

    override def traceIdGenerator(): Generator = longGenerator
    override def spanIdGenerator(): Generator = longGenerator
  }

  object Default {
    def apply(): Default = new Default()
  }


  class DoubleSizeTraceID extends Default {
    private val doubleLongGenerator = new Generator {
      override def generate(): Identifier = {
        val data = ByteBuffer.wrap(new Array[Byte](16))
        val highLong = ThreadLocalRandom.current().nextLong()
        val lowLong = ThreadLocalRandom.current().nextLong()
        data.putLong(highLong)
        data.putLong(lowLong)

        Identifier(HexCodec.toLowerHex(highLong) + HexCodec.toLowerHex(lowLong), data.array())
      }

      override def from(string: String): Identifier =  Try {
        val highPart = HexCodec.lowerHexToUnsignedLong(string.substring(0, 16))
        val lowPart = HexCodec.lowerHexToUnsignedLong(string.substring(16, 32))
        val data = ByteBuffer.allocate(16)
        data.putLong(highPart)
        data.putLong(lowPart)

        Identifier(string, data.array())
      } getOrElse(IdentityProvider.NoIdentifier)

      override def from(bytes: Array[Byte]): Identifier = Try {
        val buffer = ByteBuffer.wrap(bytes)
        val highLong = buffer.getLong
        val lowLong = buffer.getLong

        Identifier(HexCodec.toLowerHex(highLong) + HexCodec.toLowerHex(lowLong), bytes)
      } getOrElse(IdentityProvider.NoIdentifier)
    }

    override def traceIdGenerator(): Generator = doubleLongGenerator
  }

  object DoubleSizeTraceID {
    def apply(): DoubleSizeTraceID = new DoubleSizeTraceID()
  }
}