-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcats-execise-crdt4.sc
85 lines (76 loc) · 3.44 KB
/
cats-execise-crdt4.sc
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
import $ivy.`org.typelevel::cats-core:2.0.0`
import cats.instances.all._
import cats.syntax.semigroup._ //for |+|
import cats.syntax.foldable._ // for combineAll
import cats.kernel.CommutativeMonoid
trait BoundedSemiLattice[A] extends CommutativeMonoid[A] {
def combine(a1: A, a2: A): A
def empty: A
}
object BoundedSemiLattice {
implicit val intInstance: BoundedSemiLattice[Int] = new BoundedSemiLattice[Int] {
def combine(a1: Int, a2: Int): Int = a1 max a2
def empty:Int = 0
}
implicit def setInstance[A]: BoundedSemiLattice[Set[A]] = new BoundedSemiLattice[Set[A]] {
def combine(a1: Set[A], a2: Set[A]): Set[A] = a1 union a2
def empty: Set[A] = Set.empty[A]
}
}
trait GCounter[F[_,_],K, V] {
def increment(f: F[K, V])(k: K, v: V)(implicit m: CommutativeMonoid[V]): F[K, V]
def merge(f1: F[K, V], f2: F[K, V])(implicit b: BoundedSemiLattice[V]): F[K, V]
def total(f: F[K, V])(implicit m: CommutativeMonoid[V]): V
}
object GCounter {
def apply[F[_,_], K, V](implicit counter: GCounter[F, K, V]) = counter
implicit def mapInstance[K, V]: GCounter[Map, K, V] = new GCounter[Map, K, V] {
def increment(f: Map[K, V])(k: K, v: V)(implicit m: CommutativeMonoid[V]): Map[K, V] = f |+| (Map(k -> v))
def merge(f1: Map[K, V], f2: Map[K, V])(implicit b: BoundedSemiLattice[V]): Map[K, V] = f1 |+| f2
def total(f: Map[K, V])(implicit m: CommutativeMonoid[V]): V = f.values.toList.combineAll
}
}
trait KeyValueStore[F[_,_]] {
def put[K, V](f: F[K, V])(k: K, v: V): F[K, V]
def get[K, V](f: F[K, V])(k: K): Option[V]
def getOrElse[K, V](f: F[K, V])(k: K, default: V): V = get(f)(k).getOrElse(default)
def values[K, V](f: F[K, V]): List[V]
}
object KeyValueStore {
implicit val mapInstance: KeyValueStore[Map] = new KeyValueStore[Map] {
def put[K, V](f: Map[K, V])(k: K, v: V): Map[K, V] = f + (k -> v)
def get[K, V](f: Map[K, V])(k: K): Option[V] = f.get(k)
override def getOrElse[K, V](f: Map[K, V])(k: K, default: V): V = f.getOrElse(k, default)
def values[K, V](f: Map[K, V]): List[V] = f.values.toList
}
}
implicit class KvsOps[F[_,_], K, V](f: F[K, V]) {
def put(key: K, value: V)(implicit kvs: KeyValueStore[F]): F[K, V] =
kvs.put(f)(key, value)
def get(key: K)(implicit kvs: KeyValueStore[F]): Option[V] =
kvs.get(f)(key)
def getOrElse(key: K, default: V)(implicit kvs: KeyValueStore[F]): V =
kvs.getOrElse(f)(key, default)
def values(implicit kvs: KeyValueStore[F]): List[V] =
kvs.values(f)
}
//merge result is unexpected
implicit def gcounterInstance[F[_,_], K, V](implicit kvs: KeyValueStore[F], km: CommutativeMonoid[F[K, V]]) = new GCounter[F, K, V] {
def increment(f: F[K, V])(key: K, value: V)(implicit m: CommutativeMonoid[V]): F[K, V] = {
val total = f.getOrElse(key, m.empty) |+| value
f.put(key, total)
}
def merge(f1: F[K, V], f2: F[K, V])(implicit b: BoundedSemiLattice[V]): F[K, V] = {
km.combine(f1, f2)
}
def total(f: F[K, V])(implicit m: CommutativeMonoid[V]): V =
f.values.combineAll
}
val g1 = Map("a" -> 7, "b" -> 3)
val g2 = Map("a" -> 2, "b" -> 5)
val counter = gcounterInstance[Map, String, Int]
val merged = counter.merge(g1, g2)
// correct should be merged: Map[String, Int] = Map("a" -> 7, "b" -> 5)
val total = counter.total(merged)
// correct should be total: Int = 12
println(s"counter: $counter, merged: $merged, total: $total")