Files
ERitors-Scribe-Desktop/electron/database/models/Sync.ts
natreex 209dc6f85a Remove CharacterComponent and CharacterDetail components
- Deleted `CharacterComponent` and `CharacterDetail` files from the project.
- Refactored related logic to improve code maintainability and reduce redundancy.
2026-02-05 14:12:08 -05:00

1278 lines
81 KiB
TypeScript

import { getUserEncryptionKey } from "../keyManager.js";
import System from "../System.js";
import { BookSyncCompare, CompleteBook, SyncedBook, SyncedBookTools } from "./Book.js";
import { SyncedSpell, SyncedSpellTag } from "./Spell.js";
import SpellRepo, { BookSpellsTable, SyncedSpellResult } from "../repositories/spell.repo.js";
import SpellTagRepo, { BookSpellTagsTable, SyncedSpellTagResult } from "../repositories/spelltag.repo.js";
import BookRepo, { EritBooksTable, SyncedBookResult, BookToolsTable, SyncedBookToolsResult } from "../repositories/book.repository.js";
import ChapterRepo, {
BookChapterInfosTable,
BookChaptersTable,
SyncedChapterInfoResult,
SyncedChapterResult
} from "../repositories/chapter.repository.js";
import PlotPointRepository, { BookPlotPointsTable, SyncedPlotPointResult } from "../repositories/plotpoint.repository.js";
import IncidentRepository, { BookIncidentsTable, SyncedIncidentResult } from "../repositories/incident.repository.js";
import ChapterContentRepository, {
BookChapterContentTable,
SyncedChapterContentResult
} from "../repositories/chaptercontent.repository.js";
import CharacterRepo, {
BookCharactersAttributesTable,
BookCharactersTable,
SyncedCharacterAttributeResult,
SyncedCharacterResult
} from "../repositories/character.repository.js";
import LocationRepo, {
BookLocationTable,
LocationElementTable,
LocationSubElementTable,
SyncedLocationElementResult,
SyncedLocationResult,
SyncedLocationSubElementResult
} from "../repositories/location.repository.js";
import WorldRepository, {
BookWorldElementsTable,
BookWorldTable,
SyncedWorldElementResult,
SyncedWorldResult
} from "../repositories/world.repository.js";
import ActRepository, {
BookActSummariesTable,
SyncedActSummaryResult
} from "../repositories/act.repository.js";
import GuidelineRepo, {
BookAIGuideLineTable,
BookGuideLineTable,
SyncedAIGuideLineResult,
SyncedGuideLineResult
} from "../repositories/guideline.repository.js";
import IssueRepository, { BookIssuesTable, SyncedIssueResult } from "../repositories/issue.repository.js";
import { SyncedChapter, SyncedChapterContent, SyncedChapterInfo } from "./Chapter.js";
import { SyncedCharacter, SyncedCharacterAttribute } from "./Character.js";
import { SyncedLocation, SyncedLocationElement, SyncedLocationSubElement } from "./Location.js";
import { SyncedWorld, SyncedWorldElement } from "./World.js";
import { SyncedIncident } from "./Incident.js";
import { SyncedPlotPoint } from "./PlotPoint.js";
import { SyncedIssue } from "./Issue.js";
import { SyncedActSummary } from "./Act.js";
import { SyncedAIGuideLine, SyncedGuideLine } from "./GuideLine.js";
import {
SyncedSeries,
SyncedSeriesBook,
SyncedSeriesCharacter,
SyncedSeriesCharacterAttribute,
SyncedSeriesWorld,
SyncedSeriesWorldElement,
SyncedSeriesLocation,
SyncedSeriesLocationElement,
SyncedSeriesLocationSubElement,
SyncedSeriesSpell,
SyncedSeriesSpellTag
} from "./Book.js";
import SeriesRepo, {
SyncedSeriesResult,
SyncedSeriesBookResult
} from "../repositories/series.repo.js";
import SeriesCharacterRepo, {
SyncedSeriesCharacterResult,
SyncedSeriesCharacterAttributeResult
} from "../repositories/series-character.repo.js";
import SeriesWorldRepo, {
SyncedSeriesWorldResult,
SyncedSeriesWorldElementResult
} from "../repositories/series-world.repo.js";
import SeriesLocationRepo, {
SyncedSeriesLocationResult,
SyncedSeriesLocationElementResult,
SyncedSeriesLocationSubElementResult
} from "../repositories/series-location.repo.js";
import SeriesSpellRepo, {
SyncedSeriesSpellResult,
SyncedSeriesSpellTagResult
} from "../repositories/series-spell.repo.js";
/**
* Handles synchronization operations between local database and remote server.
* Provides methods to fetch, compare, and sync book data including all related entities.
*/
export default class Sync {
/**
* Retrieves a complete book with all its associated entities for synchronization.
* Decrypts all encrypted fields using the user's encryption key.
* @param userId - The unique identifier of the user
* @param syncCompareData - Object containing IDs of entities to retrieve for sync comparison
* @param lang - The language for error messages ('fr' or 'en')
* @returns A promise resolving to a CompleteBook object with all decrypted data
*/
static async getCompleteSyncBook(userId: string, syncCompareData: BookSyncCompare, lang: "fr" | "en"): Promise<CompleteBook> {
const userEncryptionKey: string = getUserEncryptionKey(userId);
const decryptedBooks: EritBooksTable[] = [];
const decryptedChapters: BookChaptersTable[] = [];
const decryptedPlotPoints: BookPlotPointsTable[] = [];
const decryptedIncidents: BookIncidentsTable[] = [];
const decryptedChapterContents: BookChapterContentTable[] = [];
const decryptedChapterInfos: BookChapterInfosTable[] = [];
const decryptedCharacters: BookCharactersTable[] = [];
const decryptedCharacterAttributes: BookCharactersAttributesTable[] = [];
const decryptedLocations: BookLocationTable[] = [];
const decryptedLocationElements: LocationElementTable[] = [];
const decryptedLocationSubElements: LocationSubElementTable[] = [];
const decryptedWorlds: BookWorldTable[] = [];
const decryptedWorldElements: BookWorldElementsTable[] = [];
const decryptedActSummaries: BookActSummariesTable[] = [];
const decryptedGuideLines: BookGuideLineTable[] = [];
const decryptedAIGuideLines: BookAIGuideLineTable[] = [];
const decryptedIssues: BookIssuesTable[] = [];
const decryptedSpells: BookSpellsTable[] = [];
const decryptedSpellTags: BookSpellTagsTable[] = [];
const actSummaryIds: string[] = syncCompareData.actSummaries;
const chapterIds: string[] = syncCompareData.chapters;
const plotPointIds: string[] = syncCompareData.plotPoints;
const incidentIds: string[] = syncCompareData.incidents;
const chapterContentIds: string[] = syncCompareData.chapterContents;
const chapterInfoIds: string[] = syncCompareData.chapterInfos;
const characterIds: string[] = syncCompareData.characters;
const characterAttributeIds: string[] = syncCompareData.characterAttributes;
const locationIds: string[] = syncCompareData.locations;
const locationElementIds: string[] = syncCompareData.locationElements;
const locationSubElementIds: string[] = syncCompareData.locationSubElements;
const worldIds: string[] = syncCompareData.worlds;
const worldElementIds: string[] = syncCompareData.worldElements;
const issueIds: string[] = syncCompareData.issues;
const spellIds: string[] = syncCompareData.spells;
const spellTagIds: string[] = syncCompareData.spellTags;
if (actSummaryIds && actSummaryIds.length > 0) {
for (const actSummaryId of actSummaryIds) {
const actSummaryResults: BookActSummariesTable[] = await ActRepository.fetchCompleteActSummaryById(actSummaryId, lang);
if (actSummaryResults.length > 0) {
const actSummaryRecord: BookActSummariesTable = actSummaryResults[0];
decryptedActSummaries.push({
...actSummaryRecord,
summary: actSummaryRecord.summary ? System.decryptDataWithUserKey(actSummaryRecord.summary, userEncryptionKey) : null
});
}
}
}
if (chapterIds && chapterIds.length > 0) {
for (const chapterId of chapterIds) {
const chapterResults: BookChaptersTable[] = await ChapterRepo.fetchCompleteChapterById(chapterId, lang);
if (chapterResults.length > 0) {
const chapterRecord: BookChaptersTable = chapterResults[0];
decryptedChapters.push({
...chapterRecord,
title: System.decryptDataWithUserKey(chapterRecord.title, userEncryptionKey)
});
}
}
}
if (plotPointIds && plotPointIds.length > 0) {
for (const plotPointId of plotPointIds) {
const plotPointResults: BookPlotPointsTable[] = await PlotPointRepository.fetchCompletePlotPointById(plotPointId, lang);
if (plotPointResults.length > 0) {
const plotPointRecord: BookPlotPointsTable = plotPointResults[0];
decryptedPlotPoints.push({
...plotPointRecord,
title: System.decryptDataWithUserKey(plotPointRecord.title, userEncryptionKey),
summary: plotPointRecord.summary ? System.decryptDataWithUserKey(plotPointRecord.summary, userEncryptionKey) : null
});
}
}
}
if (incidentIds && incidentIds.length > 0) {
for (const incidentId of incidentIds) {
const incidentResults: BookIncidentsTable[] = await IncidentRepository.fetchCompleteIncidentById(incidentId, lang);
if (incidentResults.length > 0) {
const incidentRecord: BookIncidentsTable = incidentResults[0];
decryptedIncidents.push({
...incidentRecord,
title: System.decryptDataWithUserKey(incidentRecord.title, userEncryptionKey),
summary: incidentRecord.summary ? System.decryptDataWithUserKey(incidentRecord.summary, userEncryptionKey) : null
});
}
}
}
if (chapterContentIds && chapterContentIds.length > 0) {
for (const chapterContentId of chapterContentIds) {
const chapterContentResults: BookChapterContentTable[] = await ChapterContentRepository.fetchCompleteChapterContentById(chapterContentId, lang);
if (chapterContentResults.length > 0) {
const chapterContentRecord: BookChapterContentTable = chapterContentResults[0];
decryptedChapterContents.push({
...chapterContentRecord,
content: chapterContentRecord.content ? JSON.parse(System.decryptDataWithUserKey(chapterContentRecord.content, userEncryptionKey)) : null
});
}
}
}
if (chapterInfoIds && chapterInfoIds.length > 0) {
for (const chapterInfoId of chapterInfoIds) {
const chapterInfoResults: BookChapterInfosTable[] = await ChapterRepo.fetchCompleteChapterInfoById(chapterInfoId, lang);
if (chapterInfoResults.length > 0) {
const chapterInfoRecord: BookChapterInfosTable = chapterInfoResults[0];
decryptedChapterInfos.push({
...chapterInfoRecord,
summary: chapterInfoRecord.summary ? System.decryptDataWithUserKey(chapterInfoRecord.summary, userEncryptionKey) : null,
goal: chapterInfoRecord.goal ? System.decryptDataWithUserKey(chapterInfoRecord.goal, userEncryptionKey) : null
});
}
}
}
if (characterIds && characterIds.length > 0) {
for (const characterId of characterIds) {
const characterResults: BookCharactersTable[] = await CharacterRepo.fetchCompleteCharacterById(characterId, lang);
if (characterResults.length > 0) {
const characterRecord: BookCharactersTable = characterResults[0];
decryptedCharacters.push({
...characterRecord,
first_name: System.decryptDataWithUserKey(characterRecord.first_name, userEncryptionKey),
last_name: characterRecord.last_name ? System.decryptDataWithUserKey(characterRecord.last_name, userEncryptionKey) : null,
nickname: characterRecord.nickname ? System.decryptDataWithUserKey(characterRecord.nickname, userEncryptionKey) : null,
age: characterRecord.age ? System.decryptDataWithUserKey(characterRecord.age, userEncryptionKey) : null,
gender: characterRecord.gender ? System.decryptDataWithUserKey(characterRecord.gender, userEncryptionKey) : null,
species: characterRecord.species ? System.decryptDataWithUserKey(characterRecord.species, userEncryptionKey) : null,
nationality: characterRecord.nationality ? System.decryptDataWithUserKey(characterRecord.nationality, userEncryptionKey) : null,
status: characterRecord.status ? System.decryptDataWithUserKey(characterRecord.status, userEncryptionKey) : null,
category: System.decryptDataWithUserKey(characterRecord.category, userEncryptionKey),
title: characterRecord.title ? System.decryptDataWithUserKey(characterRecord.title, userEncryptionKey) : null,
role: characterRecord.role ? System.decryptDataWithUserKey(characterRecord.role, userEncryptionKey) : null,
biography: characterRecord.biography ? System.decryptDataWithUserKey(characterRecord.biography, userEncryptionKey) : null,
history: characterRecord.history ? System.decryptDataWithUserKey(characterRecord.history, userEncryptionKey) : null,
speech_pattern: characterRecord.speech_pattern ? System.decryptDataWithUserKey(characterRecord.speech_pattern, userEncryptionKey) : null,
catchphrase: characterRecord.catchphrase ? System.decryptDataWithUserKey(characterRecord.catchphrase, userEncryptionKey) : null,
residence: characterRecord.residence ? System.decryptDataWithUserKey(characterRecord.residence, userEncryptionKey) : null,
notes: characterRecord.notes ? System.decryptDataWithUserKey(characterRecord.notes, userEncryptionKey) : null,
color: characterRecord.color ? System.decryptDataWithUserKey(characterRecord.color, userEncryptionKey) : null
});
}
}
}
if (characterAttributeIds && characterAttributeIds.length > 0) {
for (const characterAttributeId of characterAttributeIds) {
const characterAttributeResults: BookCharactersAttributesTable[] = await CharacterRepo.fetchCompleteCharacterAttributeById(characterAttributeId, lang);
if (characterAttributeResults.length > 0) {
const characterAttributeRecord: BookCharactersAttributesTable = characterAttributeResults[0];
decryptedCharacterAttributes.push({
...characterAttributeRecord,
attribute_name: System.decryptDataWithUserKey(characterAttributeRecord.attribute_name, userEncryptionKey),
attribute_value: System.decryptDataWithUserKey(characterAttributeRecord.attribute_value, userEncryptionKey)
});
}
}
}
if (locationIds && locationIds.length > 0) {
for (const locationId of locationIds) {
const locationResults: BookLocationTable[] = await LocationRepo.fetchCompleteLocationById(locationId, lang);
if (locationResults.length > 0) {
const locationRecord: BookLocationTable = locationResults[0];
decryptedLocations.push({
...locationRecord,
loc_name: System.decryptDataWithUserKey(locationRecord.loc_name, userEncryptionKey)
});
}
}
}
if (locationElementIds && locationElementIds.length > 0) {
for (const locationElementId of locationElementIds) {
const locationElementResults: LocationElementTable[] = await LocationRepo.fetchCompleteLocationElementById(locationElementId, lang);
if (locationElementResults.length > 0) {
const locationElementRecord: LocationElementTable = locationElementResults[0];
decryptedLocationElements.push({
...locationElementRecord,
element_name: System.decryptDataWithUserKey(locationElementRecord.element_name, userEncryptionKey),
element_description: locationElementRecord.element_description ? System.decryptDataWithUserKey(locationElementRecord.element_description, userEncryptionKey) : null
});
}
}
}
if (locationSubElementIds && locationSubElementIds.length > 0) {
for (const locationSubElementId of locationSubElementIds) {
const locationSubElementResults: LocationSubElementTable[] = await LocationRepo.fetchCompleteLocationSubElementById(locationSubElementId, lang);
if (locationSubElementResults.length > 0) {
const locationSubElementRecord: LocationSubElementTable = locationSubElementResults[0];
decryptedLocationSubElements.push({
...locationSubElementRecord,
sub_elem_name: System.decryptDataWithUserKey(locationSubElementRecord.sub_elem_name, userEncryptionKey),
sub_elem_description: locationSubElementRecord.sub_elem_description ? System.decryptDataWithUserKey(locationSubElementRecord.sub_elem_description, userEncryptionKey) : null
});
}
}
}
if (worldIds && worldIds.length > 0) {
for (const worldId of worldIds) {
const worldResults: BookWorldTable[] = await WorldRepository.fetchCompleteWorldById(worldId, lang);
if (worldResults.length > 0) {
const worldRecord: BookWorldTable = worldResults[0];
decryptedWorlds.push({
...worldRecord,
name: System.decryptDataWithUserKey(worldRecord.name, userEncryptionKey),
history: worldRecord.history ? System.decryptDataWithUserKey(worldRecord.history, userEncryptionKey) : null,
politics: worldRecord.politics ? System.decryptDataWithUserKey(worldRecord.politics, userEncryptionKey) : null,
economy: worldRecord.economy ? System.decryptDataWithUserKey(worldRecord.economy, userEncryptionKey) : null,
religion: worldRecord.religion ? System.decryptDataWithUserKey(worldRecord.religion, userEncryptionKey) : null,
languages: worldRecord.languages ? System.decryptDataWithUserKey(worldRecord.languages, userEncryptionKey) : null
});
}
}
}
if (worldElementIds && worldElementIds.length > 0) {
for (const worldElementId of worldElementIds) {
const worldElementResults: BookWorldElementsTable[] = await WorldRepository.fetchCompleteWorldElementById(worldElementId, lang);
if (worldElementResults.length > 0) {
const worldElementRecord: BookWorldElementsTable = worldElementResults[0];
decryptedWorldElements.push({
...worldElementRecord,
name: System.decryptDataWithUserKey(worldElementRecord.name, userEncryptionKey),
description: worldElementRecord.description ? System.decryptDataWithUserKey(worldElementRecord.description, userEncryptionKey) : null
});
}
}
}
if (issueIds && issueIds.length > 0) {
for (const issueId of issueIds) {
const issueResults: BookIssuesTable[] = await IssueRepository.fetchCompleteIssueById(issueId, lang);
if (issueResults.length > 0) {
const issueRecord: BookIssuesTable = issueResults[0];
decryptedIssues.push({
...issueRecord,
name: System.decryptDataWithUserKey(issueRecord.name, userEncryptionKey)
});
}
}
}
if (syncCompareData.guideLine) {
const guidelineResults: BookGuideLineTable[] = await GuidelineRepo.fetchBookGuideLineTable(userId, syncCompareData.id, lang);
if (guidelineResults.length > 0) {
const guidelineRecord: BookGuideLineTable = guidelineResults[0];
decryptedGuideLines.push({
...guidelineRecord,
tone: guidelineRecord.tone ? System.decryptDataWithUserKey(guidelineRecord.tone, userEncryptionKey) : null,
atmosphere: guidelineRecord.atmosphere ? System.decryptDataWithUserKey(guidelineRecord.atmosphere, userEncryptionKey) : null,
writing_style: guidelineRecord.writing_style ? System.decryptDataWithUserKey(guidelineRecord.writing_style, userEncryptionKey) : null,
themes: guidelineRecord.themes ? System.decryptDataWithUserKey(guidelineRecord.themes, userEncryptionKey) : null,
symbolism: guidelineRecord.symbolism ? System.decryptDataWithUserKey(guidelineRecord.symbolism, userEncryptionKey) : null,
motifs: guidelineRecord.motifs ? System.decryptDataWithUserKey(guidelineRecord.motifs, userEncryptionKey) : null,
narrative_voice: guidelineRecord.narrative_voice ? System.decryptDataWithUserKey(guidelineRecord.narrative_voice, userEncryptionKey) : null,
pacing: guidelineRecord.pacing ? System.decryptDataWithUserKey(guidelineRecord.pacing, userEncryptionKey) : null,
intended_audience: guidelineRecord.intended_audience ? System.decryptDataWithUserKey(guidelineRecord.intended_audience, userEncryptionKey) : null,
key_messages: guidelineRecord.key_messages ? System.decryptDataWithUserKey(guidelineRecord.key_messages, userEncryptionKey) : null
});
}
}
if (syncCompareData.aiGuideLine) {
const aiGuidelineResults: BookAIGuideLineTable[] = await GuidelineRepo.fetchBookAIGuideLine(userId, syncCompareData.id, lang);
if (aiGuidelineResults.length > 0) {
const aiGuidelineRecord: BookAIGuideLineTable = aiGuidelineResults[0];
decryptedAIGuideLines.push({
...aiGuidelineRecord,
global_resume: aiGuidelineRecord.global_resume ? System.decryptDataWithUserKey(aiGuidelineRecord.global_resume, userEncryptionKey) : null,
themes: aiGuidelineRecord.themes ? System.decryptDataWithUserKey(aiGuidelineRecord.themes, userEncryptionKey) : null,
tone: aiGuidelineRecord.tone ? System.decryptDataWithUserKey(aiGuidelineRecord.tone, userEncryptionKey) : null,
atmosphere: aiGuidelineRecord.atmosphere ? System.decryptDataWithUserKey(aiGuidelineRecord.atmosphere, userEncryptionKey) : null,
current_resume: aiGuidelineRecord.current_resume ? System.decryptDataWithUserKey(aiGuidelineRecord.current_resume, userEncryptionKey) : null
});
}
}
if (spellTagIds && spellTagIds.length > 0) {
for (const spellTagId of spellTagIds) {
const spellTagRecord: BookSpellTagsTable | null = SpellTagRepo.fetchSpellTagTableById(userId, spellTagId, lang);
if (spellTagRecord) {
decryptedSpellTags.push({
...spellTagRecord,
name: System.decryptDataWithUserKey(spellTagRecord.name, userEncryptionKey)
});
}
}
}
if (spellIds && spellIds.length > 0) {
for (const spellId of spellIds) {
const spellRecord: BookSpellsTable | null = SpellRepo.fetchSpellTableById(userId, spellId, lang);
if (spellRecord) {
decryptedSpells.push({
...spellRecord,
name: System.decryptDataWithUserKey(spellRecord.name, userEncryptionKey),
description: spellRecord.description ? System.decryptDataWithUserKey(spellRecord.description, userEncryptionKey) : null,
appearance: spellRecord.appearance ? System.decryptDataWithUserKey(spellRecord.appearance, userEncryptionKey) : null,
tags: spellRecord.tags ? System.decryptDataWithUserKey(spellRecord.tags, userEncryptionKey) : null,
power_level: spellRecord.power_level ? System.decryptDataWithUserKey(spellRecord.power_level, userEncryptionKey) : null,
components: spellRecord.components ? System.decryptDataWithUserKey(spellRecord.components, userEncryptionKey) : null,
limitations: spellRecord.limitations ? System.decryptDataWithUserKey(spellRecord.limitations, userEncryptionKey) : null,
notes: spellRecord.notes ? System.decryptDataWithUserKey(spellRecord.notes, userEncryptionKey) : null
});
}
}
}
const bookResults: EritBooksTable[] = await BookRepo.fetchCompleteBookById(syncCompareData.id, lang);
if (bookResults.length > 0) {
const bookRecord: EritBooksTable = bookResults[0];
decryptedBooks.push({
...bookRecord,
title: System.decryptDataWithUserKey(bookRecord.title, userEncryptionKey),
sub_title: bookRecord.sub_title ? System.decryptDataWithUserKey(bookRecord.sub_title, userEncryptionKey) : null,
summary: bookRecord.summary ? System.decryptDataWithUserKey(bookRecord.summary, userEncryptionKey) : null,
cover_image: bookRecord.cover_image ? System.decryptDataWithUserKey(bookRecord.cover_image, userEncryptionKey) : null
});
}
const bookToolsResult: BookToolsTable | null = BookRepo.fetchBookTools(userId, syncCompareData.id, lang);
const bookTools: BookToolsTable[] = bookToolsResult ? [bookToolsResult] : [];
return {
eritBooks: decryptedBooks,
chapters: decryptedChapters,
plotPoints: decryptedPlotPoints,
incidents: decryptedIncidents,
chapterContents: decryptedChapterContents,
chapterInfos: decryptedChapterInfos,
characters: decryptedCharacters,
characterAttributes: decryptedCharacterAttributes,
locations: decryptedLocations,
locationElements: decryptedLocationElements,
locationSubElements: decryptedLocationSubElements,
worlds: decryptedWorlds,
worldElements: decryptedWorldElements,
actSummaries: decryptedActSummaries,
guideLine: decryptedGuideLines,
aiGuideLine: decryptedAIGuideLines,
issues: decryptedIssues,
bookTools: bookTools,
spells: decryptedSpells,
spellTags: decryptedSpellTags
};
}
/**
* Synchronizes a complete book from the server to the local client database.
* Encrypts all data before storing and handles both insert and update operations.
* @param userId - The unique identifier of the user
* @param completeBook - The complete book data received from the server
* @param lang - The language for error messages ('fr' or 'en')
* @returns A promise resolving to true if sync was successful, false otherwise
*/
static async syncBookFromServerToClient(userId: string, completeBook: CompleteBook, lang: "fr" | "en"): Promise<boolean> {
const userEncryptionKey: string = getUserEncryptionKey(userId);
const serverActSummaries: BookActSummariesTable[] = completeBook.actSummaries;
const serverChapters: BookChaptersTable[] = completeBook.chapters;
const serverPlotPoints: BookPlotPointsTable[] = completeBook.plotPoints;
const serverIncidents: BookIncidentsTable[] = completeBook.incidents;
const serverChapterContents: BookChapterContentTable[] = completeBook.chapterContents;
const serverChapterInfos: BookChapterInfosTable[] = completeBook.chapterInfos;
const serverCharacters: BookCharactersTable[] = completeBook.characters;
const serverCharacterAttributes: BookCharactersAttributesTable[] = completeBook.characterAttributes;
const serverLocations: BookLocationTable[] = completeBook.locations;
const serverLocationElements: LocationElementTable[] = completeBook.locationElements;
const serverLocationSubElements: LocationSubElementTable[] = completeBook.locationSubElements;
const serverWorlds: BookWorldTable[] = completeBook.worlds;
const serverWorldElements: BookWorldElementsTable[] = completeBook.worldElements;
const serverIssues: BookIssuesTable[] = completeBook.issues;
const serverGuideLines: BookGuideLineTable[] = completeBook.guideLine;
const serverAIGuideLines: BookAIGuideLineTable[] = completeBook.aiGuideLine;
const bookId: string = completeBook.eritBooks.length > 0 ? completeBook.eritBooks[0].book_id : '';
if (serverChapters && serverChapters.length > 0) {
for (const serverChapter of serverChapters) {
const chapterExists: boolean = ChapterRepo.isChapterExist(userId, serverChapter.chapter_id, lang);
const encryptedTitle: string = System.encryptDataWithUserKey(serverChapter.title, userEncryptionKey);
if (chapterExists) {
const updateSuccessful: boolean = ChapterRepo.updateChapter(userId, serverChapter.chapter_id, encryptedTitle, serverChapter.hashed_title, serverChapter.chapter_order, serverChapter.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = ChapterRepo.insertSyncChapter(serverChapter.chapter_id, serverChapter.book_id, userId, encryptedTitle, serverChapter.hashed_title, serverChapter.words_count || 0, serverChapter.chapter_order, serverChapter.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverActSummaries && serverActSummaries.length > 0) {
for (const serverActSummary of serverActSummaries) {
const actSummaryExists: boolean = ActRepository.actSummarizeExist(userId, bookId, serverActSummary.act_index, lang);
const encryptedSummary: string = System.encryptDataWithUserKey(serverActSummary.summary ? serverActSummary.summary : '', userEncryptionKey);
if (actSummaryExists) {
const updateSuccessful: boolean = ActRepository.updateActSummary(userId, bookId, serverActSummary.act_index, encryptedSummary, serverActSummary.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = ActRepository.insertSyncActSummary(serverActSummary.act_sum_id, userId, bookId, serverActSummary.act_index, encryptedSummary, serverActSummary.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverPlotPoints && serverPlotPoints.length > 0) {
for (const serverPlotPoint of serverPlotPoints) {
const encryptedTitle: string = System.encryptDataWithUserKey(serverPlotPoint.title, userEncryptionKey);
const encryptedSummary: string = System.encryptDataWithUserKey(serverPlotPoint.summary ? serverPlotPoint.summary : '', userEncryptionKey);
const plotPointExists: boolean = PlotPointRepository.plotPointExist(userId, bookId, serverPlotPoint.plot_point_id, lang);
if (plotPointExists) {
const updateSuccessful: boolean = PlotPointRepository.updatePlotPoint(userId, bookId, serverPlotPoint.plot_point_id, encryptedTitle, serverPlotPoint.hashed_title, encryptedSummary, serverPlotPoint.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
if (!serverPlotPoint.linked_incident_id) {
return false;
}
const insertSuccessful: boolean = PlotPointRepository.insertSyncPlotPoint(serverPlotPoint.plot_point_id, encryptedTitle, serverPlotPoint.hashed_title, encryptedSummary, serverPlotPoint.linked_incident_id, serverPlotPoint.author_id, bookId, serverPlotPoint.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverIncidents && serverIncidents.length > 0) {
for (const serverIncident of serverIncidents) {
const encryptedTitle: string = System.encryptDataWithUserKey(serverIncident.title, userEncryptionKey);
const encryptedSummary: string = System.encryptDataWithUserKey(serverIncident.summary ? serverIncident.summary : '', userEncryptionKey);
const incidentExists: boolean = IncidentRepository.incidentExist(userId, bookId, serverIncident.incident_id, lang);
if (incidentExists) {
const updateSuccessful: boolean = IncidentRepository.updateIncident(userId, bookId, serverIncident.incident_id, encryptedTitle, serverIncident.hashed_title, encryptedSummary, serverIncident.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = IncidentRepository.insertSyncIncident(serverIncident.incident_id, userId, bookId, encryptedTitle, serverIncident.hashed_title, encryptedSummary, serverIncident.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverChapterContents && serverChapterContents.length > 0) {
for (const serverChapterContent of serverChapterContents) {
const chapterContentExists: boolean = ChapterContentRepository.isChapterContentExist(userId, serverChapterContent.content_id, lang);
const encryptedContent: string = System.encryptDataWithUserKey(serverChapterContent.content ? JSON.stringify(serverChapterContent.content) : '', userEncryptionKey);
if (chapterContentExists) {
const updateSuccessful: boolean = ChapterContentRepository.updateChapterContent(userId, serverChapterContent.chapter_id, serverChapterContent.version, encryptedContent, serverChapterContent.words_count, serverChapterContent.last_update);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = ChapterContentRepository.insertSyncChapterContent(serverChapterContent.content_id, serverChapterContent.chapter_id, userId, serverChapterContent.version, encryptedContent, serverChapterContent.words_count, serverChapterContent.time_on_it, serverChapterContent.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverChapterInfos && serverChapterInfos.length > 0) {
for (const serverChapterInfo of serverChapterInfos) {
const chapterInfoExists: boolean = ChapterRepo.isChapterInfoExist(userId, serverChapterInfo.chapter_id, lang);
const encryptedSummary: string = System.encryptDataWithUserKey(serverChapterInfo.summary ? serverChapterInfo.summary : '', userEncryptionKey);
const encryptedGoal: string = System.encryptDataWithUserKey(serverChapterInfo.goal ? serverChapterInfo.goal : '', userEncryptionKey);
if (chapterInfoExists) {
const updateSuccessful: boolean = ChapterRepo.updateChapterInfos(userId, serverChapterInfo.chapter_id, serverChapterInfo.act_id, bookId, serverChapterInfo.incident_id, serverChapterInfo.plot_point_id, encryptedSummary, encryptedGoal, serverChapterInfo.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = ChapterRepo.insertSyncChapterInfo(serverChapterInfo.chapter_info_id, serverChapterInfo.chapter_id, serverChapterInfo.act_id, serverChapterInfo.incident_id, serverChapterInfo.plot_point_id, bookId, serverChapterInfo.author_id, encryptedSummary, encryptedGoal, serverChapterInfo.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverCharacters && serverCharacters.length > 0) {
for (const serverCharacter of serverCharacters) {
const characterExists: boolean = CharacterRepo.isCharacterExist(userId, serverCharacter.character_id, lang);
const characterData = {
firstName: System.encryptDataWithUserKey(serverCharacter.first_name, userEncryptionKey),
lastName: System.encryptDataWithUserKey(serverCharacter.last_name ? serverCharacter.last_name : '', userEncryptionKey),
nickname: System.encryptDataWithUserKey(serverCharacter.nickname ? serverCharacter.nickname : '', userEncryptionKey),
age: System.encryptDataWithUserKey(serverCharacter.age ? serverCharacter.age : '', userEncryptionKey),
gender: System.encryptDataWithUserKey(serverCharacter.gender ? serverCharacter.gender : '', userEncryptionKey),
species: System.encryptDataWithUserKey(serverCharacter.species ? serverCharacter.species : '', userEncryptionKey),
nationality: System.encryptDataWithUserKey(serverCharacter.nationality ? serverCharacter.nationality : '', userEncryptionKey),
status: System.encryptDataWithUserKey(serverCharacter.status ? serverCharacter.status : 'alive', userEncryptionKey),
category: System.encryptDataWithUserKey(serverCharacter.category, userEncryptionKey),
title: System.encryptDataWithUserKey(serverCharacter.title ? serverCharacter.title : '', userEncryptionKey),
image: System.encryptDataWithUserKey(serverCharacter.image ? serverCharacter.image : '', userEncryptionKey),
role: System.encryptDataWithUserKey(serverCharacter.role ? serverCharacter.role : '', userEncryptionKey),
biography: System.encryptDataWithUserKey(serverCharacter.biography ? serverCharacter.biography : '', userEncryptionKey),
history: System.encryptDataWithUserKey(serverCharacter.history ? serverCharacter.history : '', userEncryptionKey),
speechPattern: System.encryptDataWithUserKey(serverCharacter.speech_pattern ? serverCharacter.speech_pattern : '', userEncryptionKey),
catchphrase: System.encryptDataWithUserKey(serverCharacter.catchphrase ? serverCharacter.catchphrase : '', userEncryptionKey),
residence: System.encryptDataWithUserKey(serverCharacter.residence ? serverCharacter.residence : '', userEncryptionKey),
notes: System.encryptDataWithUserKey(serverCharacter.notes ? serverCharacter.notes : '', userEncryptionKey),
color: System.encryptDataWithUserKey(serverCharacter.color ? serverCharacter.color : '', userEncryptionKey)
};
if (characterExists) {
const updateSuccessful: boolean = CharacterRepo.updateCharacter(userId, serverCharacter.character_id, characterData, serverCharacter.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = CharacterRepo.insertSyncCharacter(serverCharacter.character_id, bookId, userId, characterData, serverCharacter.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverCharacterAttributes && serverCharacterAttributes.length > 0) {
for (const serverCharacterAttribute of serverCharacterAttributes) {
const characterAttributeExists: boolean = CharacterRepo.isCharacterAttributeExist(userId, serverCharacterAttribute.attr_id, lang);
const encryptedAttributeName: string = System.encryptDataWithUserKey(serverCharacterAttribute.attribute_name, userEncryptionKey);
const encryptedAttributeValue: string = System.encryptDataWithUserKey(serverCharacterAttribute.attribute_value, userEncryptionKey);
if (characterAttributeExists) {
const updateSuccessful: boolean = CharacterRepo.updateCharacterAttribute(userId, serverCharacterAttribute.attr_id, encryptedAttributeName, encryptedAttributeValue, serverCharacterAttribute.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = CharacterRepo.insertSyncCharacterAttribute(serverCharacterAttribute.attr_id, serverCharacterAttribute.character_id, userId, encryptedAttributeName, encryptedAttributeValue, serverCharacterAttribute.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverLocations && serverLocations.length > 0) {
for (const serverLocation of serverLocations) {
const locationExists: boolean = LocationRepo.isLocationExist(userId, serverLocation.loc_id, lang);
const encryptedLocationName: string = System.encryptDataWithUserKey(serverLocation.loc_name, userEncryptionKey);
if (locationExists) {
const updateSuccessful: boolean = LocationRepo.updateLocationSection(userId, serverLocation.loc_id, encryptedLocationName, serverLocation.loc_original_name, serverLocation.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = LocationRepo.insertSyncLocation(serverLocation.loc_id, bookId, userId, encryptedLocationName, serverLocation.loc_original_name, serverLocation.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverLocationElements && serverLocationElements.length > 0) {
for (const serverLocationElement of serverLocationElements) {
const locationElementExists: boolean = LocationRepo.isLocationElementExist(userId, serverLocationElement.element_id, lang);
const encryptedElementName: string = System.encryptDataWithUserKey(serverLocationElement.element_name, userEncryptionKey);
const encryptedElementDescription: string = System.encryptDataWithUserKey(serverLocationElement.element_description ? serverLocationElement.element_description : '', userEncryptionKey);
if (locationElementExists) {
const updateSuccessful: boolean = LocationRepo.updateLocationElement(userId, serverLocationElement.element_id, encryptedElementName, serverLocationElement.original_name, encryptedElementDescription, serverLocationElement.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = LocationRepo.insertSyncLocationElement(serverLocationElement.element_id, serverLocationElement.location, userId, encryptedElementName, serverLocationElement.original_name, encryptedElementDescription, serverLocationElement.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverLocationSubElements && serverLocationSubElements.length > 0) {
for (const serverLocationSubElement of serverLocationSubElements) {
const locationSubElementExists: boolean = LocationRepo.isLocationSubElementExist(userId, serverLocationSubElement.sub_element_id, lang);
const encryptedSubElementName: string = System.encryptDataWithUserKey(serverLocationSubElement.sub_elem_name, userEncryptionKey);
const encryptedSubElementDescription: string = System.encryptDataWithUserKey(serverLocationSubElement.sub_elem_description ? serverLocationSubElement.sub_elem_description : '', userEncryptionKey);
if (locationSubElementExists) {
const updateSuccessful: boolean = LocationRepo.updateLocationSubElement(userId, serverLocationSubElement.sub_element_id, encryptedSubElementName, serverLocationSubElement.original_name, encryptedSubElementDescription, serverLocationSubElement.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = LocationRepo.insertSyncLocationSubElement(serverLocationSubElement.sub_element_id, serverLocationSubElement.element_id, userId, encryptedSubElementName, serverLocationSubElement.original_name, encryptedSubElementDescription, serverLocationSubElement.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverWorlds && serverWorlds.length > 0) {
for (const serverWorld of serverWorlds) {
const worldExists: boolean = WorldRepository.worldExist(userId, bookId, serverWorld.world_id, lang);
const encryptedName: string = System.encryptDataWithUserKey(serverWorld.name, userEncryptionKey);
const encryptedHistory: string = System.encryptDataWithUserKey(serverWorld.history ? serverWorld.history : '', userEncryptionKey);
const encryptedPolitics: string = System.encryptDataWithUserKey(serverWorld.politics ? serverWorld.politics : '', userEncryptionKey);
const encryptedEconomy: string = System.encryptDataWithUserKey(serverWorld.economy ? serverWorld.economy : '', userEncryptionKey);
const encryptedReligion: string = System.encryptDataWithUserKey(serverWorld.religion ? serverWorld.religion : '', userEncryptionKey);
const encryptedLanguages: string = System.encryptDataWithUserKey(serverWorld.languages ? serverWorld.languages : '', userEncryptionKey);
if (worldExists) {
const updateSuccessful: boolean = WorldRepository.updateWorld(userId, serverWorld.world_id, encryptedName, serverWorld.hashed_name, encryptedHistory, encryptedPolitics, encryptedEconomy, encryptedReligion, encryptedLanguages, serverWorld.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = WorldRepository.insertSyncWorld(serverWorld.world_id, encryptedName, serverWorld.hashed_name, userId, bookId, encryptedHistory, encryptedPolitics, encryptedEconomy, encryptedReligion, encryptedLanguages, serverWorld.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverWorldElements && serverWorldElements.length > 0) {
for (const serverWorldElement of serverWorldElements) {
const worldElementExists: boolean = WorldRepository.worldElementExist(userId, serverWorldElement.world_id, serverWorldElement.element_id, lang);
const encryptedName: string = System.encryptDataWithUserKey(serverWorldElement.name, userEncryptionKey);
const encryptedDescription: string = System.encryptDataWithUserKey(serverWorldElement.description ? serverWorldElement.description : '', userEncryptionKey);
if (worldElementExists) {
const updateSuccessful: boolean = WorldRepository.updateWorldElement(userId, serverWorldElement.element_id, encryptedName, encryptedDescription, serverWorldElement.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = WorldRepository.insertSyncWorldElement(serverWorldElement.element_id, serverWorldElement.world_id, userId, serverWorldElement.element_type, encryptedName, serverWorldElement.original_name, encryptedDescription, serverWorldElement.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverIssues && serverIssues.length > 0) {
for (const serverIssue of serverIssues) {
const issueExists: boolean = IssueRepository.issueExist(userId, bookId, serverIssue.issue_id, lang);
const encryptedName: string = System.encryptDataWithUserKey(serverIssue.name, userEncryptionKey);
if (issueExists) {
const updateSuccessful: boolean = IssueRepository.updateIssue(userId, bookId, serverIssue.issue_id, encryptedName, serverIssue.hashed_issue_name, serverIssue.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = IssueRepository.insertSyncIssue(serverIssue.issue_id, userId, bookId, encryptedName, serverIssue.hashed_issue_name, serverIssue.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverGuideLines && serverGuideLines.length > 0) {
for (const serverGuideLine of serverGuideLines) {
const guideLineExists: boolean = GuidelineRepo.guideLineExist(userId, bookId, lang);
const encryptedTone: string | null = serverGuideLine.tone ? System.encryptDataWithUserKey(serverGuideLine.tone, userEncryptionKey) : null;
const encryptedAtmosphere: string | null = serverGuideLine.atmosphere ? System.encryptDataWithUserKey(serverGuideLine.atmosphere, userEncryptionKey) : null;
const encryptedWritingStyle: string | null = serverGuideLine.writing_style ? System.encryptDataWithUserKey(serverGuideLine.writing_style, userEncryptionKey) : null;
const encryptedThemes: string | null = serverGuideLine.themes ? System.encryptDataWithUserKey(serverGuideLine.themes, userEncryptionKey) : null;
const encryptedSymbolism: string | null = serverGuideLine.symbolism ? System.encryptDataWithUserKey(serverGuideLine.symbolism, userEncryptionKey) : null;
const encryptedMotifs: string | null = serverGuideLine.motifs ? System.encryptDataWithUserKey(serverGuideLine.motifs, userEncryptionKey) : null;
const encryptedNarrativeVoice: string | null = serverGuideLine.narrative_voice ? System.encryptDataWithUserKey(serverGuideLine.narrative_voice, userEncryptionKey) : null;
const encryptedPacing: string | null = serverGuideLine.pacing ? System.encryptDataWithUserKey(serverGuideLine.pacing, userEncryptionKey) : null;
const encryptedKeyMessages: string | null = serverGuideLine.key_messages ? System.encryptDataWithUserKey(serverGuideLine.key_messages, userEncryptionKey) : null;
const encryptedIntendedAudience: string | null = serverGuideLine.intended_audience ? System.encryptDataWithUserKey(serverGuideLine.intended_audience, userEncryptionKey) : null;
if (guideLineExists) {
const updateSuccessful: boolean = GuidelineRepo.updateGuideLine(userId, bookId, encryptedTone, encryptedAtmosphere, encryptedWritingStyle, encryptedThemes, encryptedSymbolism, encryptedMotifs, encryptedNarrativeVoice, encryptedPacing, encryptedKeyMessages, encryptedIntendedAudience, serverGuideLine.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = GuidelineRepo.insertSyncGuideLine(userId, bookId, encryptedTone, encryptedAtmosphere, encryptedWritingStyle, encryptedThemes, encryptedSymbolism, encryptedMotifs, encryptedNarrativeVoice, encryptedPacing, encryptedIntendedAudience, encryptedKeyMessages, serverGuideLine.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (serverAIGuideLines && serverAIGuideLines.length > 0) {
for (const serverAIGuideLine of serverAIGuideLines) {
const aiGuideLineExists: boolean = GuidelineRepo.aiGuideLineExist(userId, bookId, lang);
const encryptedGlobalResume: string | null = serverAIGuideLine.global_resume ? System.encryptDataWithUserKey(serverAIGuideLine.global_resume, userEncryptionKey) : null;
const encryptedThemes: string | null = serverAIGuideLine.themes ? System.encryptDataWithUserKey(serverAIGuideLine.themes, userEncryptionKey) : null;
const encryptedTone: string | null = serverAIGuideLine.tone ? System.encryptDataWithUserKey(serverAIGuideLine.tone, userEncryptionKey) : null;
const encryptedAtmosphere: string | null = serverAIGuideLine.atmosphere ? System.encryptDataWithUserKey(serverAIGuideLine.atmosphere, userEncryptionKey) : null;
const encryptedCurrentResume: string | null = serverAIGuideLine.current_resume ? System.encryptDataWithUserKey(serverAIGuideLine.current_resume, userEncryptionKey) : null;
if (aiGuideLineExists) {
const updateSuccessful: boolean = GuidelineRepo.insertAIGuideLine(userId, bookId, serverAIGuideLine.narrative_type, serverAIGuideLine.dialogue_type, encryptedGlobalResume, encryptedAtmosphere, serverAIGuideLine.verbe_tense, serverAIGuideLine.langue, encryptedThemes, serverAIGuideLine.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = GuidelineRepo.insertSyncAIGuideLine(userId, bookId, encryptedGlobalResume, encryptedThemes, serverAIGuideLine.verbe_tense, serverAIGuideLine.narrative_type, serverAIGuideLine.langue, serverAIGuideLine.dialogue_type, encryptedTone, encryptedAtmosphere, encryptedCurrentResume, serverAIGuideLine.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (completeBook.bookTools && completeBook.bookTools.length > 0) {
for (const serverBookTool of completeBook.bookTools) {
const success: boolean = BookRepo.insertSyncBookTools(bookId, userId, serverBookTool.characters_enabled, serverBookTool.worlds_enabled, serverBookTool.locations_enabled, serverBookTool.spells_enabled, serverBookTool.last_update, lang);
if (!success) {
return false;
}
}
}
if (completeBook.spellTags && completeBook.spellTags.length > 0) {
for (const serverSpellTag of completeBook.spellTags) {
const spellTagExists: boolean = SpellTagRepo.isSpellTagExist(userId, serverSpellTag.tag_id, lang);
const encryptedName: string = System.encryptDataWithUserKey(serverSpellTag.name, userEncryptionKey);
if (spellTagExists) {
const updateSuccessful: boolean = SpellTagRepo.updateSyncSpellTag(userId, serverSpellTag.tag_id, encryptedName, serverSpellTag.name_hash, serverSpellTag.color, serverSpellTag.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = SpellTagRepo.insertSyncSpellTag(serverSpellTag.tag_id, serverSpellTag.book_id, userId, encryptedName, serverSpellTag.name_hash, serverSpellTag.color, serverSpellTag.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
if (completeBook.spells && completeBook.spells.length > 0) {
for (const serverSpell of completeBook.spells) {
const spellExists: boolean = SpellRepo.isSpellExist(userId, serverSpell.spell_id, lang);
const encryptedName: string = System.encryptDataWithUserKey(serverSpell.name, userEncryptionKey);
const encryptedDescription: string | null = serverSpell.description ? System.encryptDataWithUserKey(serverSpell.description, userEncryptionKey) : null;
const encryptedAppearance: string | null = serverSpell.appearance ? System.encryptDataWithUserKey(serverSpell.appearance, userEncryptionKey) : null;
const encryptedTags: string | null = serverSpell.tags ? System.encryptDataWithUserKey(serverSpell.tags, userEncryptionKey) : null;
const encryptedPowerLevel: string | null = serverSpell.power_level ? System.encryptDataWithUserKey(serverSpell.power_level, userEncryptionKey) : null;
const encryptedComponents: string | null = serverSpell.components ? System.encryptDataWithUserKey(serverSpell.components, userEncryptionKey) : null;
const encryptedLimitations: string | null = serverSpell.limitations ? System.encryptDataWithUserKey(serverSpell.limitations, userEncryptionKey) : null;
const encryptedNotes: string | null = serverSpell.notes ? System.encryptDataWithUserKey(serverSpell.notes, userEncryptionKey) : null;
if (spellExists) {
const updateSuccessful: boolean = SpellRepo.updateSyncSpell(userId, serverSpell.spell_id, encryptedName, serverSpell.name_hash, encryptedDescription, encryptedAppearance, encryptedTags, encryptedPowerLevel, encryptedComponents, encryptedLimitations, encryptedNotes, serverSpell.last_update, lang);
if (!updateSuccessful) {
return false;
}
} else {
const insertSuccessful: boolean = SpellRepo.insertSyncSpell(serverSpell.spell_id, serverSpell.book_id, userId, encryptedName, serverSpell.name_hash, encryptedDescription, encryptedAppearance, encryptedTags, encryptedPowerLevel, encryptedComponents, encryptedLimitations, encryptedNotes, serverSpell.last_update, lang);
if (!insertSuccessful) {
return false;
}
}
}
}
return true;
}
/**
* Retrieves all synced books for a user with their complete hierarchical data structure.
* Fetches all related entities (chapters, characters, locations, etc.) and organizes them by book.
* @param userId - The unique identifier of the user
* @param lang - The language for error messages ('fr' or 'en')
* @returns A promise resolving to an array of SyncedBook objects with decrypted data
*/
static async getSyncedBooks(userId: string, lang: 'fr' | 'en'): Promise<SyncedBook[]> {
const userEncryptionKey: string = getUserEncryptionKey(userId);
const [
allBooks,
allChapters,
allChapterContents,
allChapterInfos,
allCharacters,
allCharacterAttributes,
allLocations,
allLocationElements,
allLocationSubElements,
allWorlds,
allWorldElements,
allIncidents,
allPlotPoints,
allIssues,
allActSummaries,
allGuidelines,
allAIGuidelines,
allSpells,
allSpellTags
]: [
SyncedBookResult[],
SyncedChapterResult[],
SyncedChapterContentResult[],
SyncedChapterInfoResult[],
SyncedCharacterResult[],
SyncedCharacterAttributeResult[],
SyncedLocationResult[],
SyncedLocationElementResult[],
SyncedLocationSubElementResult[],
SyncedWorldResult[],
SyncedWorldElementResult[],
SyncedIncidentResult[],
SyncedPlotPointResult[],
SyncedIssueResult[],
SyncedActSummaryResult[],
SyncedGuideLineResult[],
SyncedAIGuideLineResult[],
SyncedSpellResult[],
SyncedSpellTagResult[]
] = await Promise.all([
BookRepo.fetchSyncedBooks(userId, lang),
ChapterRepo.fetchSyncedChapters(userId, lang),
ChapterContentRepository.fetchSyncedChapterContents(userId, lang),
ChapterRepo.fetchSyncedChapterInfos(userId, lang),
CharacterRepo.fetchSyncedCharacters(userId, lang),
CharacterRepo.fetchSyncedCharacterAttributes(userId, lang),
LocationRepo.fetchSyncedLocations(userId, lang),
LocationRepo.fetchSyncedLocationElements(userId, lang),
LocationRepo.fetchSyncedLocationSubElements(userId, lang),
WorldRepository.fetchSyncedWorlds(userId, lang),
WorldRepository.fetchSyncedWorldElements(userId, lang),
IncidentRepository.fetchSyncedIncidents(userId, lang),
PlotPointRepository.fetchSyncedPlotPoints(userId, lang),
IssueRepository.fetchSyncedIssues(userId, lang),
ActRepository.fetchSyncedActSummaries(userId, lang),
GuidelineRepo.fetchSyncedGuideLine(userId, lang),
GuidelineRepo.fetchSyncedAIGuideLine(userId, lang),
SpellRepo.fetchSyncedSpells(userId, lang),
SpellTagRepo.fetchSyncedSpellTags(userId, lang)
]);
return allBooks.map((bookRecord: SyncedBookResult): SyncedBook => {
const currentBookId: string = bookRecord.book_id;
const bookChapters: SyncedChapter[] = allChapters
.filter((chapterRecord: SyncedChapterResult): boolean => chapterRecord.book_id === currentBookId)
.map((chapterRecord: SyncedChapterResult): SyncedChapter => {
const currentChapterId: string = chapterRecord.chapter_id;
const chapterContents: SyncedChapterContent[] = allChapterContents
.filter((contentRecord: SyncedChapterContentResult): boolean => contentRecord.chapter_id === currentChapterId)
.map((contentRecord: SyncedChapterContentResult): SyncedChapterContent => ({
id: contentRecord.content_id,
lastUpdate: contentRecord.last_update
}));
const chapterInfoRecord: SyncedChapterInfoResult | undefined = allChapterInfos.find((infoRecord: SyncedChapterInfoResult): boolean => infoRecord.chapter_id === currentChapterId);
const chapterInfo: SyncedChapterInfo | null = chapterInfoRecord ? {
id: chapterInfoRecord.chapter_info_id,
lastUpdate: chapterInfoRecord.last_update
} : null;
return {
id: currentChapterId,
name: System.decryptDataWithUserKey(chapterRecord.title, userEncryptionKey),
lastUpdate: chapterRecord.last_update,
contents: chapterContents,
info: chapterInfo
};
});
const bookCharacters: SyncedCharacter[] = allCharacters
.filter((characterRecord: SyncedCharacterResult): boolean => characterRecord.book_id === currentBookId)
.map((characterRecord: SyncedCharacterResult): SyncedCharacter => {
const currentCharacterId: string = characterRecord.character_id;
const characterAttributes: SyncedCharacterAttribute[] = allCharacterAttributes
.filter((attributeRecord: SyncedCharacterAttributeResult): boolean => attributeRecord.character_id === currentCharacterId)
.map((attributeRecord: SyncedCharacterAttributeResult): SyncedCharacterAttribute => ({
id: attributeRecord.attr_id,
name: System.decryptDataWithUserKey(attributeRecord.attribute_name, userEncryptionKey),
lastUpdate: attributeRecord.last_update
}));
return {
id: currentCharacterId,
name: System.decryptDataWithUserKey(characterRecord.first_name, userEncryptionKey),
lastUpdate: characterRecord.last_update,
attributes: characterAttributes
};
});
const bookLocations: SyncedLocation[] = allLocations
.filter((locationRecord: SyncedLocationResult): boolean => locationRecord.book_id === currentBookId)
.map((locationRecord: SyncedLocationResult): SyncedLocation => {
const currentLocationId: string = locationRecord.loc_id;
const locationElements: SyncedLocationElement[] = allLocationElements
.filter((elementRecord: SyncedLocationElementResult): boolean => elementRecord.location === currentLocationId)
.map((elementRecord: SyncedLocationElementResult): SyncedLocationElement => {
const currentElementId: string = elementRecord.element_id;
const locationSubElements: SyncedLocationSubElement[] = allLocationSubElements
.filter((subElementRecord: SyncedLocationSubElementResult): boolean => subElementRecord.element_id === currentElementId)
.map((subElementRecord: SyncedLocationSubElementResult): SyncedLocationSubElement => ({
id: subElementRecord.sub_element_id,
name: System.decryptDataWithUserKey(subElementRecord.sub_elem_name, userEncryptionKey),
lastUpdate: subElementRecord.last_update
}));
return {
id: currentElementId,
name: System.decryptDataWithUserKey(elementRecord.element_name, userEncryptionKey),
lastUpdate: elementRecord.last_update,
subElements: locationSubElements
};
});
return {
id: currentLocationId,
name: System.decryptDataWithUserKey(locationRecord.loc_name, userEncryptionKey),
lastUpdate: locationRecord.last_update,
elements: locationElements
};
});
const bookWorlds: SyncedWorld[] = allWorlds
.filter((worldRecord: SyncedWorldResult): boolean => worldRecord.book_id === currentBookId)
.map((worldRecord: SyncedWorldResult): SyncedWorld => {
const currentWorldId: string = worldRecord.world_id;
const worldElements: SyncedWorldElement[] = allWorldElements
.filter((worldElementRecord: SyncedWorldElementResult): boolean => worldElementRecord.world_id === currentWorldId)
.map((worldElementRecord: SyncedWorldElementResult): SyncedWorldElement => ({
id: worldElementRecord.element_id,
name: System.decryptDataWithUserKey(worldElementRecord.name, userEncryptionKey),
lastUpdate: worldElementRecord.last_update
}));
return {
id: currentWorldId,
name: System.decryptDataWithUserKey(worldRecord.name, userEncryptionKey),
lastUpdate: worldRecord.last_update,
elements: worldElements
};
});
const bookIncidents: SyncedIncident[] = allIncidents
.filter((incidentRecord: SyncedIncidentResult): boolean => incidentRecord.book_id === currentBookId)
.map((incidentRecord: SyncedIncidentResult): SyncedIncident => ({
id: incidentRecord.incident_id,
name: System.decryptDataWithUserKey(incidentRecord.title, userEncryptionKey),
lastUpdate: incidentRecord.last_update
}));
const bookPlotPoints: SyncedPlotPoint[] = allPlotPoints
.filter((plotPointRecord: SyncedPlotPointResult): boolean => plotPointRecord.book_id === currentBookId)
.map((plotPointRecord: SyncedPlotPointResult): SyncedPlotPoint => ({
id: plotPointRecord.plot_point_id,
name: System.decryptDataWithUserKey(plotPointRecord.title, userEncryptionKey),
lastUpdate: plotPointRecord.last_update
}));
const bookIssues: SyncedIssue[] = allIssues
.filter((issueRecord: SyncedIssueResult): boolean => issueRecord.book_id === currentBookId)
.map((issueRecord: SyncedIssueResult): SyncedIssue => ({
id: issueRecord.issue_id,
name: System.decryptDataWithUserKey(issueRecord.name, userEncryptionKey),
lastUpdate: issueRecord.last_update
}));
const bookActSummaries: SyncedActSummary[] = allActSummaries
.filter((actSummaryRecord: SyncedActSummaryResult): boolean => actSummaryRecord.book_id === currentBookId)
.map((actSummaryRecord: SyncedActSummaryResult): SyncedActSummary => ({
id: actSummaryRecord.act_sum_id,
lastUpdate: actSummaryRecord.last_update
}));
const guidelineRecord: SyncedGuideLineResult | undefined = allGuidelines.find((guidelineItem: SyncedGuideLineResult): boolean => guidelineItem.book_id === currentBookId);
const bookGuideLine: SyncedGuideLine | null = guidelineRecord ? {
lastUpdate: guidelineRecord.last_update
} : null;
const aiGuidelineRecord: SyncedAIGuideLineResult | undefined = allAIGuidelines.find((aiGuidelineItem: SyncedAIGuideLineResult): boolean => aiGuidelineItem.book_id === currentBookId);
const bookAIGuideLine: SyncedAIGuideLine | null = aiGuidelineRecord ? {
lastUpdate: aiGuidelineRecord.last_update
} : null;
const bookToolsQuery: SyncedBookToolsResult | null = BookRepo.fetchSyncedBookTools(userId, currentBookId, lang);
const bookTools: SyncedBookTools | null = bookToolsQuery ? {
lastUpdate: bookToolsQuery.last_update,
charactersEnabled: bookToolsQuery.characters_enabled === 1,
worldsEnabled: bookToolsQuery.worlds_enabled === 1,
locationsEnabled: bookToolsQuery.locations_enabled === 1,
spellsEnabled: bookToolsQuery.spells_enabled === 1
} : null;
const bookSpells: SyncedSpell[] = allSpells
.filter((spellRecord: SyncedSpellResult): boolean => spellRecord.book_id === currentBookId)
.map((spellRecord: SyncedSpellResult): SyncedSpell => ({
id: spellRecord.spell_id,
name: System.decryptDataWithUserKey(spellRecord.name, userEncryptionKey),
lastUpdate: spellRecord.last_update
}));
const bookSpellTags: SyncedSpellTag[] = allSpellTags
.filter((spellTagRecord: SyncedSpellTagResult): boolean => spellTagRecord.book_id === currentBookId)
.map((spellTagRecord: SyncedSpellTagResult): SyncedSpellTag => ({
id: spellTagRecord.tag_id,
name: System.decryptDataWithUserKey(spellTagRecord.name, userEncryptionKey),
lastUpdate: spellTagRecord.last_update
}));
return {
id: currentBookId,
type: bookRecord.type,
title: System.decryptDataWithUserKey(bookRecord.title, userEncryptionKey),
subTitle: bookRecord.sub_title ? System.decryptDataWithUserKey(bookRecord.sub_title, userEncryptionKey) : null,
lastUpdate: bookRecord.last_update,
chapters: bookChapters,
characters: bookCharacters,
locations: bookLocations,
worlds: bookWorlds,
incidents: bookIncidents,
plotPoints: bookPlotPoints,
issues: bookIssues,
actSummaries: bookActSummaries,
guideLine: bookGuideLine,
aiGuideLine: bookAIGuideLine,
bookTools: bookTools,
spells: bookSpells,
spellTags: bookSpellTags
};
});
}
// ===== SERIES SYNC METHODS =====
/**
* Retrieves all series for the current user with lightweight structure for sync comparison.
* Returns synced series with nested characters, worlds, locations, spells, and spell tags.
* All encrypted fields are decrypted before return.
* @param userId - The unique identifier of the user
* @param lang - The language for error messages ('fr' or 'en')
* @returns An array of synced series with all nested entities
*/
static getSyncedSeries(userId: string, lang: 'fr' | 'en'): SyncedSeries[] {
const userEncryptionKey: string = getUserEncryptionKey(userId);
const allSeries: SyncedSeriesResult[] = SeriesRepo.fetchSyncedSeries(userId, lang);
const allSeriesBooks: SyncedSeriesBookResult[] = SeriesRepo.fetchSyncedSeriesBooks(userId, lang);
const allCharacters: SyncedSeriesCharacterResult[] = SeriesCharacterRepo.fetchSyncedSeriesCharacters(userId, lang);
const allCharacterAttributes: SyncedSeriesCharacterAttributeResult[] = SeriesCharacterRepo.fetchSyncedSeriesCharacterAttributes(userId, lang);
const allWorlds: SyncedSeriesWorldResult[] = SeriesWorldRepo.fetchSyncedSeriesWorlds(userId, lang);
const allWorldElements: SyncedSeriesWorldElementResult[] = SeriesWorldRepo.fetchSyncedSeriesWorldElements(userId, lang);
const allLocations: SyncedSeriesLocationResult[] = SeriesLocationRepo.fetchSyncedSeriesLocations(userId, lang);
const allLocationElements: SyncedSeriesLocationElementResult[] = SeriesLocationRepo.fetchSyncedSeriesLocationElements(userId, lang);
const allLocationSubElements: SyncedSeriesLocationSubElementResult[] = SeriesLocationRepo.fetchSyncedSeriesLocationSubElements(userId, lang);
const allSpells: SyncedSeriesSpellResult[] = SeriesSpellRepo.fetchSyncedSeriesSpells(userId, lang);
const allSpellTags: SyncedSeriesSpellTagResult[] = SeriesSpellRepo.fetchSyncedSeriesSpellTags(userId, lang);
return allSeries.map((series: SyncedSeriesResult): SyncedSeries => {
const seriesId: string = series.series_id;
// Map series books
const books: SyncedSeriesBook[] = allSeriesBooks
.filter((sb: SyncedSeriesBookResult): boolean => sb.series_id === seriesId)
.map((sb: SyncedSeriesBookResult): SyncedSeriesBook => ({
bookId: sb.book_id,
order: sb.book_order,
lastUpdate: sb.last_update
}));
// Map characters with attributes
const characters: SyncedSeriesCharacter[] = allCharacters
.filter((c: SyncedSeriesCharacterResult): boolean => c.series_id === seriesId)
.map((c: SyncedSeriesCharacterResult): SyncedSeriesCharacter => ({
id: c.character_id,
name: System.decryptDataWithUserKey(c.first_name, userEncryptionKey),
lastUpdate: c.last_update,
attributes: allCharacterAttributes
.filter((a: SyncedSeriesCharacterAttributeResult): boolean => a.character_id === c.character_id)
.map((a: SyncedSeriesCharacterAttributeResult): SyncedSeriesCharacterAttribute => ({
id: a.attr_id,
name: System.decryptDataWithUserKey(a.attribute_name, userEncryptionKey),
lastUpdate: a.last_update
}))
}));
// Map worlds with elements
const worlds: SyncedSeriesWorld[] = allWorlds
.filter((w: SyncedSeriesWorldResult): boolean => w.series_id === seriesId)
.map((w: SyncedSeriesWorldResult): SyncedSeriesWorld => ({
id: w.world_id,
name: System.decryptDataWithUserKey(w.name, userEncryptionKey),
lastUpdate: w.last_update,
elements: allWorldElements
.filter((e: SyncedSeriesWorldElementResult): boolean => e.world_id === w.world_id)
.map((e: SyncedSeriesWorldElementResult): SyncedSeriesWorldElement => ({
id: e.element_id,
name: System.decryptDataWithUserKey(e.name, userEncryptionKey),
lastUpdate: e.last_update
}))
}));
// Map locations with elements and sub-elements
const locations: SyncedSeriesLocation[] = allLocations
.filter((l: SyncedSeriesLocationResult): boolean => l.series_id === seriesId)
.map((l: SyncedSeriesLocationResult): SyncedSeriesLocation => ({
id: l.loc_id,
name: System.decryptDataWithUserKey(l.loc_name, userEncryptionKey),
lastUpdate: l.last_update,
elements: allLocationElements
.filter((e: SyncedSeriesLocationElementResult): boolean => e.location_id === l.loc_id)
.map((e: SyncedSeriesLocationElementResult): SyncedSeriesLocationElement => ({
id: e.element_id,
name: System.decryptDataWithUserKey(e.element_name, userEncryptionKey),
lastUpdate: e.last_update,
subElements: allLocationSubElements
.filter((se: SyncedSeriesLocationSubElementResult): boolean => se.element_id === e.element_id)
.map((se: SyncedSeriesLocationSubElementResult): SyncedSeriesLocationSubElement => ({
id: se.sub_element_id,
name: System.decryptDataWithUserKey(se.sub_elem_name, userEncryptionKey),
lastUpdate: se.last_update
}))
}))
}));
// Map spells
const spells: SyncedSeriesSpell[] = allSpells
.filter((s: SyncedSeriesSpellResult): boolean => s.series_id === seriesId)
.map((s: SyncedSeriesSpellResult): SyncedSeriesSpell => ({
id: s.spell_id,
name: System.decryptDataWithUserKey(s.name, userEncryptionKey),
lastUpdate: s.last_update
}));
// Map spell tags
const spellTags: SyncedSeriesSpellTag[] = allSpellTags
.filter((t: SyncedSeriesSpellTagResult): boolean => t.series_id === seriesId)
.map((t: SyncedSeriesSpellTagResult): SyncedSeriesSpellTag => ({
id: t.tag_id,
name: System.decryptDataWithUserKey(t.name, userEncryptionKey),
lastUpdate: t.last_update
}));
return {
id: seriesId,
name: System.decryptDataWithUserKey(series.name, userEncryptionKey),
description: series.description
? System.decryptDataWithUserKey(series.description, userEncryptionKey)
: null,
lastUpdate: series.last_update,
books,
characters,
worlds,
locations,
spells,
spellTags
};
});
}
}