summaryrefslogtreecommitdiff
path: root/test/files/pos/depmet_implicit_oopsla_session_2.scala
blob: 5c3b78e3f595ecb3eabbd35ed08ac3159ae1916e (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
object Sessions {
  def ?[T <: AnyRef](implicit w: T): w.type = w

  // session states
  sealed case class Stop()
  sealed case class In[-Data, +Cont](recv: Data => Cont)
  sealed case class Out[+Data, +Cont](data: Data, cont: Cont)

  // the type theory of communicating sessions:

  // an instance of type Session[S]{type Dual=D} is evidence that S and D are duals
  // such a value witnesses this fact by describing how to compose an instance of S with an instance of D (through the run method)
  trait Session[S] { type Self = S
    type Dual
    type HasDual[D] = Session[Self]{type Dual=D}
    def run(self: Self, dual: Dual): Unit
  }

  // friendly interface to the theory
  def runSession[S, D: Session[S]#HasDual](session: S, dual: D) =
    ?[Session[S]#HasDual[D]].run(session, dual)

  // facts in the theory:

  // ------------------------[StopDual]
  // Stop is the dual of Stop
  implicit object StopDual extends Session[Stop] {
    type Dual = Stop

    def run(self: Self, dual: Dual): Unit = {}
  }

  //            CD is the dual of Cont
  // -------------------------------------------[InDual]
  // Out[Data, CD] is the dual of In[Data, Cont]
  implicit def InDual[Data, Cont](implicit cont: Session[Cont]) = new Session[In[Data, Cont]] {
    type Dual = Out[Data, cont.Dual]

    def run(self: Self, dual: Dual): Unit =
      cont.run(self.recv(dual.data), dual.cont)
  }

  //            CD is the dual of Cont
  // -------------------------------------------[OutDual]
  // In[Data, CD] is the dual of Out[Data, Cont]
  implicit def OutDual[Data, Cont](implicit cont: Session[Cont]) = new Session[Out[Data, Cont]] {
    type Dual = In[Data, cont.Dual]

    def run(self: Self, dual: Dual): Unit =
      cont.run(self.cont, dual.recv(self.data))
  }

  // a concrete session
  def addServer =
    In{x: Int =>
    In{y: Int => System.out.println("Thinking")
    Out(x+y,
    Stop())}}

  def addClient =
    Out(3,
    Out(4, { System.out.println("Waiting")
    In{z: Int => System.out.println(z)
    Stop()}}))

  def myRun = runSession(addServer, addClient)
}

/* future improvements:


  // def runSession[S, D](p: S, dp: D)(implicit s: Session[S]#HasDual[D]) =
  //   s.run(p, dp)
  //
  // def runSession[S, D](p: S, dp: D)(implicit s: Session[S]{type Dual=D}) =
  //   s.run(p, dp)

  // TODO: can we relax the ordering restrictions on dependencies so that we can write
  //  one possibility: graph of dependencies between arguments must be acyclic
  // def runSession[S](p: S, dp: s.Dual)(implicit s: Session[S]) =
  //   s.run(p, dp)
  // to emphasise similarity of type parameters and implicit arguments:
  // def runSession[S][val s: Session[S]](p: S, dp: s.Dual) =
  //   s.run(p, dp)


*/