summaryrefslogtreecommitdiff
path: root/test/files/run/repl-serialization.scala
blob: 8bc0dd3a8b821e68c65b3986b8e1de2ece1cbb9b (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
import java.io._

import scala.reflect.io.AbstractFile
import scala.tools.nsc.Settings
import scala.tools.nsc.interpreter.IMain
import scala.tools.nsc.util._
import scala.reflect.internal.util.AbstractFileClassLoader

object Test {
  def main(args: Array[String]) {
    run()
  }

  def run(): Unit = {
    val settings = new Settings()
    settings.Yreplclassbased.value = true
    settings.usejavacp.value = true

    var imain: IMain = null
    object extract extends ((AnyRef) => Unit) with Serializable {
      var value: AnyRef = null

      def apply(a: AnyRef) = value = a
    }

    val code =
      """val x = {println("  evaluating x"); 0 }
        |def getX() = x
        |class U extends Serializable { println("constructing U"); val x = 0 ; override def toString = "U" }
        |lazy val y = {println("  evaluating y"); 0 }
        |class D; val z = {println("  evaluating z"); 0}; val zz = {println("  evaluating zz"); 0}
        |object O extends Serializable { val apply = {println("  evaluating O"); 0} }
        |class A(i: Int) { println("  constructing A") }
        |type AA = A
        |val u = new U()
        |extract(() => new AA(x + getX() + y + z + zz + O.apply + u.x))
      """.stripMargin

    imain = IMain(settings)
    println("== evaluating lines")
    imain.directBind("extract", "(AnyRef => Unit)", extract)
    code.lines.foreach(imain.interpret)

    val virtualFile: AbstractFile = extract.value.getClass.getClassLoader.asInstanceOf[AbstractFileClassLoader].root
    val newLoader = new AbstractFileClassLoader(virtualFile, getClass.getClassLoader)

    def deserializeInNewLoader(string: Array[Byte]): AnyRef = {
      val bis = new ByteArrayInputStream(string)
      val in = new ObjectInputStream(bis) {
        override def resolveClass(desc: ObjectStreamClass) = Class.forName(desc.getName, false, newLoader)
      }
      in.readObject()
    }
    def serialize(o: AnyRef): Array[Byte] = {
      val bos = new ByteArrayOutputStream()
      val out = new ObjectOutputStream(bos)
      out.writeObject(o)
      out.close()
      bos.toByteArray
    }
    println("== evaluating lambda")
    extract.value.asInstanceOf[() => Any].apply()
    println("== reconstituting into a fresh classloader")
    val reconstituted = deserializeInNewLoader(serialize(extract.value)).asInstanceOf[() => Any]
    println("== evaluating reconstituted lambda")
    reconstituted.apply() // should not print("evaluating x") a second time
  }
}