From ff052c46ed848dceedc6c83497dc94cda40a84b8 Mon Sep 17 00:00:00 2001 From: ramil Date: Tue, 31 Oct 2023 13:25:05 +0100 Subject: [PATCH] make ImmutableLinkedHashMap serializable --- .../libs/json/JsObjectSerializationSpec.scala | 40 +++++++++++++++++++ .../libs/json/ImmutableLinkedHashMap.scala | 5 ++- .../libs/json/ImmutableLinkedHashMap.scala | 3 +- .../scala/play/api/libs/json/JsValue.scala | 1 + 4 files changed, 47 insertions(+), 2 deletions(-) create mode 100644 play-json/jvm/src/test/scala/play/api/libs/json/JsObjectSerializationSpec.scala diff --git a/play-json/jvm/src/test/scala/play/api/libs/json/JsObjectSerializationSpec.scala b/play-json/jvm/src/test/scala/play/api/libs/json/JsObjectSerializationSpec.scala new file mode 100644 index 000000000..d908be278 --- /dev/null +++ b/play-json/jvm/src/test/scala/play/api/libs/json/JsObjectSerializationSpec.scala @@ -0,0 +1,40 @@ +/* + * Copyright (C) from 2022 The Play Framework Contributors , 2011-2021 Lightbend Inc. + */ + +package play.api.libs.json + +import org.specs2.mutable._ +import play.api.libs.json.Json._ + +import java.io.{ ByteArrayInputStream, ByteArrayOutputStream, ObjectInputStream, ObjectOutputStream } + +class JsObjectSerializationSpec extends Specification { + "JsObject" should { + "serialize/deserialize correctly" in { + val originalObj = Json.obj( + "field1" -> 123, + "field2" -> "abc", + "field3" -> JsNull, + "obj" -> Json.obj("field1" -> 234), + "arr" -> JsArray( + Seq( + JsString("abc"), + JsNumber(123), + JsBoolean(true), + JsNull, + Json.obj("field1" -> 345) + ) + ) + ) + + val bos = new ByteArrayOutputStream() + val out = new ObjectOutputStream(bos) + out.writeObject(originalObj) + + val bis = new ByteArrayInputStream(bos.toByteArray) + val in = new ObjectInputStream(bis) + in.readObject().asInstanceOf[JsObject].must(beEqualTo(originalObj)) + } + } +} diff --git a/play-json/shared/src/main/scala-2.13+/play/api/libs/json/ImmutableLinkedHashMap.scala b/play-json/shared/src/main/scala-2.13+/play/api/libs/json/ImmutableLinkedHashMap.scala index dae99d481..ce4a39543 100644 --- a/play-json/shared/src/main/scala-2.13+/play/api/libs/json/ImmutableLinkedHashMap.scala +++ b/play-json/shared/src/main/scala-2.13+/play/api/libs/json/ImmutableLinkedHashMap.scala @@ -13,7 +13,10 @@ import scala.collection.mutable /** * Wraps a Java LinkedHashMap as a Scala immutable.Map. */ -private[json] class ImmutableLinkedHashMap[A, +B](underlying: JLinkedHashMap[A, B]) extends AbstractMap[A, B] { +@SerialVersionUID(-2338626292552177485L) +private[json] class ImmutableLinkedHashMap[A, +B](underlying: JLinkedHashMap[A, B]) + extends AbstractMap[A, B] + with Serializable { override def get(key: A): Option[B] = Option(underlying.get(key)) diff --git a/play-json/shared/src/main/scala-2.13-/play/api/libs/json/ImmutableLinkedHashMap.scala b/play-json/shared/src/main/scala-2.13-/play/api/libs/json/ImmutableLinkedHashMap.scala index 17b316fa9..a39925ad0 100644 --- a/play-json/shared/src/main/scala-2.13-/play/api/libs/json/ImmutableLinkedHashMap.scala +++ b/play-json/shared/src/main/scala-2.13-/play/api/libs/json/ImmutableLinkedHashMap.scala @@ -17,10 +17,11 @@ import scala.collection.mutable.ArrayBuffer /** * Wraps a Java LinkedHashMap as a Scala immutable.Map. */ +@SerialVersionUID(-2338626292552177485L) private[json] class ImmutableLinkedHashMap[A, +B](underlying: JLinkedHashMap[A, B]) extends AbstractMap[A, B] with Map[A, B] - with MapLike[A, B, ImmutableLinkedHashMap[A, B]] { + with MapLike[A, B, ImmutableLinkedHashMap[A, B]] with Serializable { override def get(key: A): Option[B] = Option(underlying.get(key)) diff --git a/play-json/shared/src/main/scala/play/api/libs/json/JsValue.scala b/play-json/shared/src/main/scala/play/api/libs/json/JsValue.scala index 5bacb0488..fa06b4977 100644 --- a/play-json/shared/src/main/scala/play/api/libs/json/JsValue.scala +++ b/play-json/shared/src/main/scala/play/api/libs/json/JsValue.scala @@ -33,6 +33,7 @@ object JsValue { * Represents a Json null value. */ case object JsNull extends JsValue { + @transient implicit val reads: Reads[JsNull.type] = Reads[JsNull.type] { case JsNull => JsSuccess(JsNull) case _ => JsError("error.expected.null")