/engine/src/main/java/org/terasology/recording/RecordAndReplaySerializer.java

http://github.com/MovingBlocks/Terasology · Java · 189 lines · 140 code · 17 blank · 32 comment · 0 complexity · 4ae751635bf240570d1bb1d96a919ce3 MD5 · raw file

  1. /*
  2. * Copyright 2018 MovingBlocks
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package org.terasology.recording;
  17. import com.google.gson.Gson;
  18. import com.google.gson.GsonBuilder;
  19. import com.google.gson.JsonElement;
  20. import com.google.gson.JsonParser;
  21. import com.google.gson.reflect.TypeToken;
  22. import com.google.gson.stream.JsonWriter;
  23. import org.slf4j.Logger;
  24. import org.slf4j.LoggerFactory;
  25. import org.terasology.engine.module.ModuleManager;
  26. import org.terasology.engine.paths.PathManager;
  27. import org.terasology.entitySystem.entity.EntityManager;
  28. import org.terasology.math.geom.Vector3f;
  29. import org.terasology.reflection.TypeRegistry;
  30. import java.io.FileWriter;
  31. import java.io.FileReader;
  32. import java.lang.reflect.Type;
  33. import java.util.ArrayList;
  34. import java.util.HashMap;
  35. import java.util.Map;
  36. /**
  37. * Responsible for serializing and saving every Recording data.
  38. */
  39. public final class RecordAndReplaySerializer {
  40. private static final Logger logger = LoggerFactory.getLogger(RecordAndReplaySerializer.class);
  41. private static final String EVENT_DIR = "/events";
  42. private static final String JSON = ".json";
  43. private static final String FILE_AMOUNT = "/file_amount" + JSON;
  44. private static final String STATE_EVENT_POSITION = "/state_event_position" + JSON;
  45. private static final String DIRECTION_ORIGIN_LIST = "/direction_origin_list" + JSON;
  46. private RecordedEventStore recordedEventStore;
  47. private RecordAndReplayUtils recordAndReplayUtils;
  48. private CharacterStateEventPositionMap characterStateEventPositionMap;
  49. private DirectionAndOriginPosRecorderList directionAndOriginPosRecorderList;
  50. private RecordedEventSerializer recordedEventSerializer;
  51. public RecordAndReplaySerializer(EntityManager manager, RecordedEventStore store,
  52. RecordAndReplayUtils recordAndReplayUtils,
  53. CharacterStateEventPositionMap characterStateEventPositionMap,
  54. DirectionAndOriginPosRecorderList directionAndOriginPosRecorderList,
  55. ModuleManager moduleManager, TypeRegistry typeRegistry) {
  56. this.recordedEventStore = store;
  57. this.recordAndReplayUtils = recordAndReplayUtils;
  58. this.characterStateEventPositionMap = characterStateEventPositionMap;
  59. this.directionAndOriginPosRecorderList = directionAndOriginPosRecorderList;
  60. this.recordedEventSerializer = new RecordedEventSerializer(manager, moduleManager, typeRegistry);
  61. }
  62. /**
  63. * Serialize the recorded data.
  64. */
  65. public void serializeRecordAndReplayData() {
  66. String recordingPath = PathManager.getInstance().getRecordingPath(recordAndReplayUtils.getGameTitle()).toString();
  67. serializeRecordedEvents(recordingPath);
  68. Gson gson = new GsonBuilder().create();
  69. serializeFileAmount(gson, recordingPath);
  70. serializeCharacterStateEventPositionMap(gson, recordingPath);
  71. serializeAttackEventExtraRecorder(gson, recordingPath);
  72. }
  73. /**
  74. * Serialize RecordedEvents.
  75. * @param recordingPath path where the data should be saved.
  76. */
  77. public void serializeRecordedEvents(String recordingPath) {
  78. String filepath = recordingPath + EVENT_DIR + recordAndReplayUtils.getFileCount() + JSON;
  79. recordAndReplayUtils.setFileAmount(recordAndReplayUtils.getFileAmount() + 1);
  80. recordAndReplayUtils.setFileCount(recordAndReplayUtils.getFileCount() + 1);
  81. recordedEventSerializer.serializeRecordedEvents(recordedEventStore.popEvents(), filepath);
  82. logger.info("RecordedEvents Serialization completed!");
  83. }
  84. /**
  85. * Deserialize recorded data.
  86. */
  87. public void deserializeRecordAndReplayData() {
  88. String recordingPath = PathManager.getInstance().getRecordingPath(recordAndReplayUtils.getGameTitle()).toString();
  89. deserializeRecordedEvents(recordingPath);
  90. Gson gson = new GsonBuilder().create();
  91. deserializeFileAmount(gson, recordingPath);
  92. deserializeCharacterStateEventPositionMap(gson, recordingPath);
  93. deserializeAttackEventExtraRecorder(gson, recordingPath);
  94. }
  95. /**
  96. * Deserialize RecordedEvents.
  97. * @param recordingPath path where the data was saved.
  98. */
  99. void deserializeRecordedEvents(String recordingPath) {
  100. String filepath = recordingPath + EVENT_DIR + recordAndReplayUtils.getFileCount() + JSON;
  101. recordAndReplayUtils.setFileCount(recordAndReplayUtils.getFileCount() + 1);
  102. recordedEventStore.setEvents(recordedEventSerializer.deserializeRecordedEvents(filepath));
  103. logger.info("RecordedEvents Deserialization completed!");
  104. }
  105. private void serializeFileAmount(Gson gson, String recordingPath) {
  106. try {
  107. JsonWriter writer = new JsonWriter(new FileWriter(recordingPath + FILE_AMOUNT));
  108. gson.toJson(recordAndReplayUtils.getFileAmount(), Integer.class, writer);
  109. writer.close();
  110. logger.info("File Amount Serialization completed!");
  111. } catch (Exception e) {
  112. logger.error("Error while serializing file amount:", e);
  113. }
  114. }
  115. private void deserializeFileAmount(Gson gson, String recordingPath) {
  116. try (FileReader fileReader = new FileReader(recordingPath + FILE_AMOUNT)){
  117. JsonParser parser = new JsonParser();
  118. JsonElement jsonElement = parser.parse(fileReader);
  119. Type typeOfCount = new TypeToken<Integer>() { }.getType();
  120. recordAndReplayUtils.setFileAmount(gson.fromJson(jsonElement, typeOfCount));
  121. logger.info("File Amount Deserialization completed!");
  122. } catch (Exception e) {
  123. logger.error("Error while deserializing file amount:", e);
  124. }
  125. }
  126. private void serializeCharacterStateEventPositionMap(Gson gson, String recordingPath) {
  127. try {
  128. JsonWriter writer = new JsonWriter(new FileWriter(recordingPath + STATE_EVENT_POSITION));
  129. gson.toJson(characterStateEventPositionMap.getIdToData(), HashMap.class, writer);
  130. writer.close();
  131. characterStateEventPositionMap.reset();
  132. logger.info("CharacterStateEvent positions Serialization completed!");
  133. } catch (Exception e) {
  134. logger.error("Error while serializing CharacterStateEvent positions:", e);
  135. }
  136. }
  137. private void deserializeCharacterStateEventPositionMap(Gson gson, String recordingPath) {
  138. try (FileReader fileReader = new FileReader(recordingPath + STATE_EVENT_POSITION)) {
  139. JsonParser parser = new JsonParser();
  140. JsonElement jsonElement = parser.parse(fileReader);
  141. Type typeOfHashMap = new TypeToken<HashMap<Integer, Vector3f[]>>() { }.getType();
  142. Map<Integer, Vector3f[]> previousMap = gson.fromJson(jsonElement, typeOfHashMap);
  143. characterStateEventPositionMap.setIdToData(previousMap);
  144. logger.info("CharacterStateEvent positions Deserialization completed!");
  145. } catch (Exception e) {
  146. logger.error("Error while deserializing CharacterStateEvent positions:", e);
  147. }
  148. }
  149. private void serializeAttackEventExtraRecorder(Gson gson, String recordingPath) {
  150. try {
  151. JsonWriter writer = new JsonWriter(new FileWriter(recordingPath + DIRECTION_ORIGIN_LIST));
  152. gson.toJson(directionAndOriginPosRecorderList.getList(), ArrayList.class, writer);
  153. writer.close();
  154. directionAndOriginPosRecorderList.reset();
  155. logger.info("AttackEvent extras serialization completed!");
  156. } catch (Exception e) {
  157. logger.error("Error while serializing AttackEvent extras:", e);
  158. }
  159. }
  160. private void deserializeAttackEventExtraRecorder(Gson gson, String recordingPath) {
  161. try (FileReader fileReader = new FileReader(recordingPath + DIRECTION_ORIGIN_LIST)) {
  162. JsonParser parser = new JsonParser();
  163. JsonElement jsonElement = parser.parse(fileReader);
  164. Type type = new TypeToken<ArrayList<DirectionAndOriginPosRecorder>>() {}.getType();
  165. ArrayList<DirectionAndOriginPosRecorder> list = gson.fromJson(jsonElement, type);
  166. directionAndOriginPosRecorderList.setList(list);
  167. logger.info("AttackEvent extras deserialization completed!");
  168. } catch (Exception e) {
  169. logger.error("Error while deserializing AttackEvent extras:", e);
  170. }
  171. }
  172. }