@firestone-hs/replay-parser
Version:
This library was generated with [Angular CLI](https://github.com/angular/angular-cli) version 8.2.9.
246 lines • 39.2 kB
JavaScript
import { Injectable } from '@angular/core';
import { Map } from 'immutable';
import { Observable } from 'rxjs';
import { Game } from '../models/game/game';
import { ActionParserConfig, GameHistoryItem } from '../models/models';
import { XmlParserService } from './xml-parser.service';
import * as i0 from "@angular/core";
import * as i1 from "./all-cards.service";
import * as i2 from "./gamepipeline/action-parser.service";
import * as i3 from "./gamepipeline/turn-parser.service";
import * as i4 from "./image-preloader.service";
import * as i5 from "./entitiespipeline/game-population.service";
import * as i6 from "./entitiespipeline/game-state-parser.service";
import * as i7 from "./gamepipeline/game-initializer.service";
import * as i8 from "./gamepipeline/active-player-parser.service";
import * as i9 from "./gamepipeline/active-spell-parser.service";
import * as i10 from "./gamepipeline/targets-parser.service";
import * as i11 from "./gamepipeline/mulligan-parser.service";
import * as i12 from "./gamepipeline/end-game-parser.service";
import * as i13 from "./gamepipeline/narrator.service";
import * as i14 from "./state-processor.service";
const SMALL_PAUSE = 15;
export class GameParserService {
constructor(allCards, actionParser, turnParser, imagePreloader, gamePopulationService, gameStateParser, gameInitializer, activePlayerParser, activeSpellParser, targetsParser, mulliganParser, endGameParser, narrator, stateProcessor) {
this.allCards = allCards;
this.actionParser = actionParser;
this.turnParser = turnParser;
this.imagePreloader = imagePreloader;
this.gamePopulationService = gamePopulationService;
this.gameStateParser = gameStateParser;
this.gameInitializer = gameInitializer;
this.activePlayerParser = activePlayerParser;
this.activeSpellParser = activeSpellParser;
this.targetsParser = targetsParser;
this.mulliganParser = mulliganParser;
this.endGameParser = endGameParser;
this.narrator = narrator;
this.stateProcessor = stateProcessor;
}
async parse(replayAsString, options, config = new ActionParserConfig()) {
const start = Date.now();
this.cancelled = false;
if (this.processingTimeout) {
clearTimeout(this.processingTimeout);
this.processingTimeout = undefined;
}
if (!this.allCards.getCards()?.length) {
await this.allCards.initializeCardsDb();
this.logPerf('Retrieved cards DB, parsing replay', start);
}
const iterator = this.createGamePipeline(replayAsString, start, options, config);
return Observable.create(observer => {
this.buildObservableFunction(observer, iterator);
});
}
cancelProcessing() {
this.cancelled = true;
clearTimeout(this.processingTimeout);
}
buildObservableFunction(observer, iterator) {
// // console.log('calling next iteration');
try {
const itValue = iterator.next();
// // console.log('calling next obersable', itValue, itValue.value);
observer.next([itValue.value[0], itValue.value[2], itValue.done]);
if (!itValue.done && !this.cancelled) {
this.processingTimeout = setTimeout(() => this.buildObservableFunction(observer, iterator), itValue.value[1]);
}
}
catch (e) {
console.error('[game-parser] Exception in buildObservableFunction', e);
}
}
*createGamePipeline(replayAsString, start, options, config) {
if (!replayAsString || replayAsString.length == 0) {
return [null, SMALL_PAUSE, 'Invalid XML replay'];
}
// console.log('[game-parser] preparing entity / acrd ID mapping');
let entityCardId = Map([]);
const fullEntityIdCardIdMatcher = new RegExp(/id="(.*?)" cardID="(.*?)"/g);
const fullEntityMatchResult = replayAsString.match(fullEntityIdCardIdMatcher);
for (let match of fullEntityMatchResult) {
const result = new RegExp(/id="(.*?)" cardID="(.*?)"/g).exec(match);
if (result) {
entityCardId = entityCardId.set(parseInt(result[1]), result[2]);
}
}
const showEntityIdCardIdMatcher = new RegExp(/cardID="(.*?)" entity="(.*?)"/g);
const showEntityMatchResult = replayAsString.match(showEntityIdCardIdMatcher);
for (let match of showEntityMatchResult) {
// // console.log("updating with show', result", copy);
const result = new RegExp(/cardID="(.*?)" entity="(.*?)"/g).exec(match);
if (result) {
// // console.log('result', result);
entityCardId = entityCardId.set(parseInt(result[2]), result[1]);
}
}
// console.log('[game-parser] mapping done', entityCardId.size);
// Do the parsing turn by turn
// let history: readonly HistoryItem[];
const xmlParsingIterator = new XmlParserService().parseXml(replayAsString);
let game = Game.createGame({});
let counter = 0;
while (true) {
const itValue = xmlParsingIterator.next();
const history = itValue.value;
// console.debug('[game-parser] parsing for', counter, 'with history length', history.length);
if (!history || itValue.done) {
// console.debug('[game-parser] history parsing over', itValue);
break;
}
if (history[0] instanceof GameHistoryItem) {
const gameHistory = history[0];
game = Object.assign(game, {
buildNumber: gameHistory.buildNumber,
formatType: gameHistory.formatType,
gameType: gameHistory.gameType,
scenarioID: gameHistory.scenarioID,
});
// console.log('[game-parser] assign meta data to game', game);
}
// Battlegrounds tutorial
if (game.scenarioID === 3539) {
// console.log('[game-parser] Battlegrounds tutorial not supported, returning');
return [null, SMALL_PAUSE, 'Batllegrounds tutorial is not supported'];
}
// Preload the images we'll need early on
// const preloadIterator = this.imagePreloader.preloadImages(history);
// while (true) {
// const itValue = preloadIterator.next();
// if (itValue.done) {
// break;
// }
// }
// console.log('[game-parser] will initNewEntities', game, history, entityCardId.toJS());
let entities = this.gamePopulationService.initNewEntities(game, history, entityCardId);
// console.log('[game-parser] initNewEntities', entities.size);
if (game.turns.size === 0) {
game = this.gameInitializer.initializePlayers(game, entities);
game = this.gameStateParser.updateEntitiesUntilMulliganState(game, entities, history);
entities = game.entitiesBeforeMulligan;
// // console.log('game after populateEntitiesUntilMulliganState', game, game.turns.toJS());
}
game = this.turnParser.createTurns(game, history);
// // console.log('game after turn creation', game.turns.size);
game = this.actionParser.parseActions(game, entities, history, config);
// // console.log(
// 'entity 150 parseActions',
// game.getLatestParsedState().get(150) &&
// game
// .getLatestParsedState()
// .get(150)
// .tags.toJS(),
// );
// // console.log('game after action pasring', game.getLatestParsedState().toJS());
if (game.turns.size > 0) {
game = this.activePlayerParser.parseActivePlayerForLastTurn(game);
// // console.log(
// 'entity 150 parseActivePlayerForLastTurn',
// game.getLatestParsedState().get(150) &&
// game
// .getLatestParsedState()
// .get(150)
// .tags.toJS(),
// );
// // console.log('game after parseActivePlayer', game, game.turns.toJS());
game = this.activeSpellParser.parseActiveSpellForLastTurn(game);
// // console.log(
// 'entity 150 parseActiveSpellForLastTurn',
// game.getLatestParsedState().get(150) &&
// game
// .getLatestParsedState()
// .get(150)
// .tags.toJS(),
// );
// // console.log('game after parseActiveSpell', game, game.turns.toJS());
game = this.targetsParser.parseTargetsForLastTurn(game);
// // console.log(
// 'entity 150 parseTargetsForLastTurn',
// game.getLatestParsedState().get(150) &&
// game
// .getLatestParsedState()
// .get(150)
// .tags.toJS(),
// );
// // console.log('game after parseTargets', game, game.turns.toJS());
if (game.turns.size === 1) {
game = this.mulliganParser.affectMulligan(game);
}
// // console.log('game after affectMulligan', game, game.turns.toJS());
game = this.endGameParser.parseEndGame(game);
// // console.log('game after parseEndGame', game, game.turns.toJS());
game = this.narrator.populateActionTextForLastTurn(game);
// // console.log('game after populateActionText', game, game.turns.toJS());
game = this.narrator.createGameStoryForLastTurn(game);
// // console.log(
// 'entity 150 createGameStoryForLastTurn',
// game.getLatestParsedState().get(150) &&
// game
// .getLatestParsedState()
// .get(150)
// .tags.toJS(),
// );
// // console.log('game after createGameStory', game, game.turns.toJS());
// if (counter === 4) {
// counter++;
// // console.log('returning', counter);
// return [game, SMALL_PAUSE, 'Rendering game state'];
// }
// counter++;
// // console.log('moving on', counter);
// if (game.turns.size === 33) {
// // console.log(
// 'entities at end of turn',
// game.getLatestParsedState().toJS(),
// game.getLatestParsedState().get(507),
// );
// }
yield [game, SMALL_PAUSE, 'Parsed turn ' + counter++];
}
else {
// if (counter++ === 3) {
// counter++;
// // // console.log('returning', counter, game.entities.get(73), game.entities.get(74));
// return [game, SMALL_PAUSE, 'Rendering game state'];
// }
// counter++;
}
}
// console.log('parsing done, returning');
return [game, SMALL_PAUSE, 'Rendering game state'];
}
logPerf(what, start, result) {
// console.log('[perf] ', what, 'done after ', Date.now() - start, 'ms');
return result;
}
}
GameParserService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.1", ngImport: i0, type: GameParserService, deps: [{ token: i1.AllCardsService }, { token: i2.ActionParserService }, { token: i3.TurnParserService }, { token: i4.ImagePreloaderService }, { token: i5.GamePopulationService }, { token: i6.GameStateParserService }, { token: i7.GameInitializerService }, { token: i8.ActivePlayerParserService }, { token: i9.ActiveSpellParserService }, { token: i10.TargetsParserService }, { token: i11.MulliganParserService }, { token: i12.EndGameParserService }, { token: i13.NarratorService }, { token: i14.StateProcessorService }], target: i0.ɵɵFactoryTarget.Injectable });
GameParserService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.0.1", ngImport: i0, type: GameParserService, providedIn: 'root' });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.1", ngImport: i0, type: GameParserService, decorators: [{
type: Injectable,
args: [{
providedIn: 'root',
}]
}], ctorParameters: function () { return [{ type: i1.AllCardsService }, { type: i2.ActionParserService }, { type: i3.TurnParserService }, { type: i4.ImagePreloaderService }, { type: i5.GamePopulationService }, { type: i6.GameStateParserService }, { type: i7.GameInitializerService }, { type: i8.ActivePlayerParserService }, { type: i9.ActiveSpellParserService }, { type: i10.TargetsParserService }, { type: i11.MulliganParserService }, { type: i12.EndGameParserService }, { type: i13.NarratorService }, { type: i14.StateProcessorService }]; } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"game-parser.service.js","sourceRoot":"","sources":["../../../../../projects/replay-parser/src/lib/services/game-parser.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { Map } from 'immutable';\r\nimport { Observable } from 'rxjs';\r\nimport { Game } from '../models/game/game';\r\nimport { HistoryItem } from '../models/history/history-item';\r\nimport { ActionParserConfig, GameHistoryItem } from '../models/models';\r\nimport { AllCardsService } from './all-cards.service';\r\nimport { GamePopulationService } from './entitiespipeline/game-population.service';\r\nimport { GameStateParserService } from './entitiespipeline/game-state-parser.service';\r\nimport { ActionParserService } from './gamepipeline/action-parser.service';\r\nimport { ActivePlayerParserService } from './gamepipeline/active-player-parser.service';\r\nimport { ActiveSpellParserService } from './gamepipeline/active-spell-parser.service';\r\nimport { EndGameParserService } from './gamepipeline/end-game-parser.service';\r\nimport { GameInitializerService } from './gamepipeline/game-initializer.service';\r\nimport { MulliganParserService } from './gamepipeline/mulligan-parser.service';\r\nimport { NarratorService } from './gamepipeline/narrator.service';\r\nimport { TargetsParserService } from './gamepipeline/targets-parser.service';\r\nimport { TurnParserService } from './gamepipeline/turn-parser.service';\r\nimport { ImagePreloaderService } from './image-preloader.service';\r\nimport { StateProcessorService } from './state-processor.service';\r\nimport { XmlParserService } from './xml-parser.service';\r\n\r\nconst SMALL_PAUSE = 15;\r\n\r\n@Injectable({\r\n\tprovidedIn: 'root',\r\n})\r\nexport class GameParserService {\r\n\tconstructor(\r\n\t\tprivate allCards: AllCardsService,\r\n\t\tprivate actionParser: ActionParserService,\r\n\t\tprivate turnParser: TurnParserService,\r\n\t\tprivate imagePreloader: ImagePreloaderService,\r\n\t\tprivate gamePopulationService: GamePopulationService,\r\n\t\tprivate gameStateParser: GameStateParserService,\r\n\t\tprivate gameInitializer: GameInitializerService,\r\n\t\tprivate activePlayerParser: ActivePlayerParserService,\r\n\t\tprivate activeSpellParser: ActiveSpellParserService,\r\n\t\tprivate targetsParser: TargetsParserService,\r\n\t\tprivate mulliganParser: MulliganParserService,\r\n\t\tprivate endGameParser: EndGameParserService,\r\n\t\tprivate narrator: NarratorService,\r\n\t\tprivate stateProcessor: StateProcessorService,\r\n\t) {}\r\n\tprivate cancelled: boolean;\r\n\tprivate processingTimeout;\r\n\r\n\tpublic async parse(\r\n\t\treplayAsString: string,\r\n\t\toptions?: TechnicalParsingOptions,\r\n\t\tconfig: ActionParserConfig = new ActionParserConfig(),\r\n\t): Promise<Observable<[Game, string, boolean]>> {\r\n\t\tconst start = Date.now();\r\n\t\tthis.cancelled = false;\r\n\t\tif (this.processingTimeout) {\r\n\t\t\tclearTimeout(this.processingTimeout);\r\n\t\t\tthis.processingTimeout = undefined;\r\n\t\t}\r\n\r\n\t\tif (!this.allCards.getCards()?.length) {\r\n\t\t\tawait this.allCards.initializeCardsDb();\r\n\t\t\tthis.logPerf('Retrieved cards DB, parsing replay', start);\r\n\t\t}\r\n\r\n\t\tconst iterator: IterableIterator<[Game, number, string]> = this.createGamePipeline(\r\n\t\t\treplayAsString,\r\n\t\t\tstart,\r\n\t\t\toptions,\r\n\t\t\tconfig,\r\n\t\t);\r\n\t\treturn Observable.create(observer => {\r\n\t\t\tthis.buildObservableFunction(observer, iterator);\r\n\t\t});\r\n\t}\r\n\r\n\tpublic cancelProcessing(): void {\r\n\t\tthis.cancelled = true;\r\n\t\tclearTimeout(this.processingTimeout);\r\n\t}\r\n\r\n\tprivate buildObservableFunction(observer, iterator: IterableIterator<[Game, number, string]>) {\r\n\t\t// // console.log('calling next iteration');\r\n\t\ttry {\r\n\t\t\tconst itValue = iterator.next();\r\n\t\t\t// // console.log('calling next obersable', itValue, itValue.value);\r\n\t\t\tobserver.next([itValue.value[0], itValue.value[2], itValue.done]);\r\n\t\t\tif (!itValue.done && !this.cancelled) {\r\n\t\t\t\tthis.processingTimeout = setTimeout(\r\n\t\t\t\t\t() => this.buildObservableFunction(observer, iterator),\r\n\t\t\t\t\titValue.value[1],\r\n\t\t\t\t);\r\n\t\t\t}\r\n\t\t} catch (e) {\r\n\t\t\tconsole.error('[game-parser] Exception in buildObservableFunction', e);\r\n\t\t}\r\n\t}\r\n\r\n\tprivate *createGamePipeline(\r\n\t\treplayAsString: string,\r\n\t\tstart: number,\r\n\t\toptions: TechnicalParsingOptions,\r\n\t\tconfig: ActionParserConfig,\r\n\t): IterableIterator<[Game, number, string]> {\r\n\t\tif (!replayAsString || replayAsString.length == 0) {\r\n\t\t\treturn [null, SMALL_PAUSE, 'Invalid XML replay'];\r\n\t\t}\r\n\r\n\t\t// console.log('[game-parser] preparing entity / acrd ID mapping');\r\n\t\tlet entityCardId: Map<number, string> = Map([]);\r\n\t\tconst fullEntityIdCardIdMatcher = new RegExp(/id=\"(.*?)\" cardID=\"(.*?)\"/g);\r\n\t\tconst fullEntityMatchResult = replayAsString.match(fullEntityIdCardIdMatcher);\r\n\t\tfor (let match of fullEntityMatchResult) {\r\n\t\t\tconst result = new RegExp(/id=\"(.*?)\" cardID=\"(.*?)\"/g).exec(match);\r\n\t\t\tif (result) {\r\n\t\t\t\tentityCardId = entityCardId.set(parseInt(result[1]), result[2]);\r\n\t\t\t}\r\n\t\t}\r\n\t\tconst showEntityIdCardIdMatcher = new RegExp(/cardID=\"(.*?)\" entity=\"(.*?)\"/g);\r\n\t\tconst showEntityMatchResult = replayAsString.match(showEntityIdCardIdMatcher);\r\n\t\tfor (let match of showEntityMatchResult) {\r\n\t\t\t// // console.log(\"updating with show', result\", copy);\r\n\t\t\tconst result = new RegExp(/cardID=\"(.*?)\" entity=\"(.*?)\"/g).exec(match);\r\n\t\t\tif (result) {\r\n\t\t\t\t// // console.log('result', result);\r\n\t\t\t\tentityCardId = entityCardId.set(parseInt(result[2]), result[1]);\r\n\t\t\t}\r\n\t\t}\r\n\t\t// console.log('[game-parser] mapping done', entityCardId.size);\r\n\r\n\t\t// Do the parsing turn by turn\r\n\t\t// let history: readonly HistoryItem[];\r\n\t\tconst xmlParsingIterator: IterableIterator<readonly HistoryItem[]> = new XmlParserService().parseXml(\r\n\t\t\treplayAsString,\r\n\t\t);\r\n\t\tlet game: Game = Game.createGame({} as Game);\r\n\t\tlet counter = 0;\r\n\t\twhile (true) {\r\n\t\t\tconst itValue = xmlParsingIterator.next();\r\n\t\t\tconst history: readonly HistoryItem[] = itValue.value;\r\n\t\t\t// console.debug('[game-parser] parsing for', counter, 'with history length', history.length);\r\n\r\n\t\t\tif (!history || itValue.done) {\r\n\t\t\t\t// console.debug('[game-parser] history parsing over', itValue);\r\n\t\t\t\tbreak;\r\n\t\t\t}\r\n\r\n\t\t\tif (history[0] instanceof GameHistoryItem) {\r\n\t\t\t\tconst gameHistory: GameHistoryItem = history[0] as GameHistoryItem;\r\n\t\t\t\tgame = Object.assign(game, {\r\n\t\t\t\t\tbuildNumber: gameHistory.buildNumber,\r\n\t\t\t\t\tformatType: gameHistory.formatType,\r\n\t\t\t\t\tgameType: gameHistory.gameType,\r\n\t\t\t\t\tscenarioID: gameHistory.scenarioID,\r\n\t\t\t\t} as Game);\r\n\t\t\t\t// console.log('[game-parser] assign meta data to game', game);\r\n\t\t\t}\r\n\r\n\t\t\t// Battlegrounds tutorial\r\n\t\t\tif (game.scenarioID === 3539) {\r\n\t\t\t\t// console.log('[game-parser] Battlegrounds tutorial not supported, returning');\r\n\t\t\t\treturn [null, SMALL_PAUSE, 'Batllegrounds tutorial is not supported'];\r\n\t\t\t}\r\n\r\n\t\t\t// Preload the images we'll need early on\r\n\t\t\t// const preloadIterator = this.imagePreloader.preloadImages(history);\r\n\t\t\t// while (true) {\r\n\t\t\t// \tconst itValue = preloadIterator.next();\r\n\t\t\t// \tif (itValue.done) {\r\n\t\t\t// \t\tbreak;\r\n\t\t\t// \t}\r\n\t\t\t// }\r\n\r\n\t\t\t// console.log('[game-parser] will initNewEntities', game, history, entityCardId.toJS());\r\n\t\t\tlet entities = this.gamePopulationService.initNewEntities(game, history, entityCardId);\r\n\t\t\t// console.log('[game-parser] initNewEntities', entities.size);\r\n\t\t\tif (game.turns.size === 0) {\r\n\t\t\t\tgame = this.gameInitializer.initializePlayers(game, entities);\r\n\t\t\t\tgame = this.gameStateParser.updateEntitiesUntilMulliganState(game, entities, history);\r\n\t\t\t\tentities = game.entitiesBeforeMulligan;\r\n\t\t\t\t// // console.log('game after populateEntitiesUntilMulliganState', game, game.turns.toJS());\r\n\t\t\t}\r\n\r\n\t\t\tgame = this.turnParser.createTurns(game, history);\r\n\t\t\t// // console.log('game after turn creation', game.turns.size);\r\n\t\t\tgame = this.actionParser.parseActions(game, entities, history, config);\r\n\t\t\t// // console.log(\r\n\t\t\t// \t'entity 150 parseActions',\r\n\t\t\t// \tgame.getLatestParsedState().get(150) &&\r\n\t\t\t// \t\tgame\r\n\t\t\t// \t\t\t.getLatestParsedState()\r\n\t\t\t// \t\t\t.get(150)\r\n\t\t\t// \t\t\t.tags.toJS(),\r\n\t\t\t// );\r\n\t\t\t// // console.log('game after action pasring', game.getLatestParsedState().toJS());\r\n\t\t\tif (game.turns.size > 0) {\r\n\t\t\t\tgame = this.activePlayerParser.parseActivePlayerForLastTurn(game);\r\n\t\t\t\t// // console.log(\r\n\t\t\t\t// \t'entity 150 parseActivePlayerForLastTurn',\r\n\t\t\t\t// \tgame.getLatestParsedState().get(150) &&\r\n\t\t\t\t// \t\tgame\r\n\t\t\t\t// \t\t\t.getLatestParsedState()\r\n\t\t\t\t// \t\t\t.get(150)\r\n\t\t\t\t// \t\t\t.tags.toJS(),\r\n\t\t\t\t// );\r\n\t\t\t\t// // console.log('game after parseActivePlayer', game, game.turns.toJS());\r\n\t\t\t\tgame = this.activeSpellParser.parseActiveSpellForLastTurn(game);\r\n\t\t\t\t// // console.log(\r\n\t\t\t\t// \t'entity 150 parseActiveSpellForLastTurn',\r\n\t\t\t\t// \tgame.getLatestParsedState().get(150) &&\r\n\t\t\t\t// \t\tgame\r\n\t\t\t\t// \t\t\t.getLatestParsedState()\r\n\t\t\t\t// \t\t\t.get(150)\r\n\t\t\t\t// \t\t\t.tags.toJS(),\r\n\t\t\t\t// );\r\n\t\t\t\t// // console.log('game after parseActiveSpell', game, game.turns.toJS());\r\n\t\t\t\tgame = this.targetsParser.parseTargetsForLastTurn(game);\r\n\t\t\t\t// // console.log(\r\n\t\t\t\t// \t'entity 150 parseTargetsForLastTurn',\r\n\t\t\t\t// \tgame.getLatestParsedState().get(150) &&\r\n\t\t\t\t// \t\tgame\r\n\t\t\t\t// \t\t\t.getLatestParsedState()\r\n\t\t\t\t// \t\t\t.get(150)\r\n\t\t\t\t// \t\t\t.tags.toJS(),\r\n\t\t\t\t// );\r\n\t\t\t\t// // console.log('game after parseTargets', game, game.turns.toJS());\r\n\t\t\t\tif (game.turns.size === 1) {\r\n\t\t\t\t\tgame = this.mulliganParser.affectMulligan(game);\r\n\t\t\t\t}\r\n\t\t\t\t// // console.log('game after affectMulligan', game, game.turns.toJS());\r\n\t\t\t\tgame = this.endGameParser.parseEndGame(game);\r\n\t\t\t\t// // console.log('game after parseEndGame', game, game.turns.toJS());\r\n\t\t\t\tgame = this.narrator.populateActionTextForLastTurn(game);\r\n\t\t\t\t// // console.log('game after populateActionText', game, game.turns.toJS());\r\n\t\t\t\tgame = this.narrator.createGameStoryForLastTurn(game);\r\n\t\t\t\t// // console.log(\r\n\t\t\t\t// \t'entity 150 createGameStoryForLastTurn',\r\n\t\t\t\t// \tgame.getLatestParsedState().get(150) &&\r\n\t\t\t\t// \t\tgame\r\n\t\t\t\t// \t\t\t.getLatestParsedState()\r\n\t\t\t\t// \t\t\t.get(150)\r\n\t\t\t\t// \t\t\t.tags.toJS(),\r\n\t\t\t\t// );\r\n\t\t\t\t// // console.log('game after createGameStory', game, game.turns.toJS());\r\n\t\t\t\t// if (counter === 4) {\r\n\t\t\t\t// \tcounter++;\r\n\t\t\t\t// \t// console.log('returning', counter);\r\n\t\t\t\t// \treturn [game, SMALL_PAUSE, 'Rendering game state'];\r\n\t\t\t\t// }\r\n\t\t\t\t// counter++;\r\n\t\t\t\t// // console.log('moving on', counter);\r\n\t\t\t\t// if (game.turns.size === 33) {\r\n\t\t\t\t// \t// console.log(\r\n\t\t\t\t// \t\t'entities at end of turn',\r\n\t\t\t\t// \t\tgame.getLatestParsedState().toJS(),\r\n\t\t\t\t// \t\tgame.getLatestParsedState().get(507),\r\n\t\t\t\t// \t);\r\n\t\t\t\t// }\r\n\r\n\t\t\t\tyield [game, SMALL_PAUSE, 'Parsed turn ' + counter++];\r\n\t\t\t} else {\r\n\t\t\t\t// if (counter++ === 3) {\r\n\t\t\t\t// \tcounter++;\r\n\t\t\t\t// \t// // console.log('returning', counter, game.entities.get(73), game.entities.get(74));\r\n\t\t\t\t// \treturn [game, SMALL_PAUSE, 'Rendering game state'];\r\n\t\t\t\t// }\r\n\t\t\t\t// counter++;\r\n\t\t\t}\r\n\t\t}\r\n\t\t// console.log('parsing done, returning');\r\n\t\treturn [game, SMALL_PAUSE, 'Rendering game state'];\r\n\t}\r\n\r\n\tprivate logPerf<T>(what: string, start: number, result?: T): T {\r\n\t\t// console.log('[perf] ', what, 'done after ', Date.now() - start, 'ms');\r\n\t\treturn result;\r\n\t}\r\n}\r\n\r\nexport interface GameProcessingStep {\r\n\tgame: Game;\r\n\tshouldBubble: boolean;\r\n}\r\n\r\nexport interface TechnicalParsingOptions {\r\n\treadonly shouldYield: number;\r\n\treadonly skipUi: boolean;\r\n}\r\n"]}