From 3cbaef9a4ba22f3bcd6da06affc29765dc69f3fd Mon Sep 17 00:00:00 2001 From: Manuel Martin Date: Wed, 15 May 2024 14:07:34 +0200 Subject: [PATCH] Backup Spoke project assets --- package.json | 2 +- src/api/backup.ts | 58 ++++++++++++++++++++++++++++++++++++---- src/api/reticulum-api.ts | 1 + 3 files changed, 55 insertions(+), 6 deletions(-) diff --git a/package.json b/package.json index cfdac0e..b208949 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "hubs-backup-tool", "productName": "Hubs Backup Tool", - "version": "1.0.4", + "version": "1.0.5", "description": "Tool for backing up your assets from a Hubs instance", "main": ".webpack/main", "scripts": { diff --git a/src/api/backup.ts b/src/api/backup.ts index 54577ae..3f0ff75 100644 --- a/src/api/backup.ts +++ b/src/api/backup.ts @@ -168,17 +168,64 @@ async function backupBlenderProjects( } } +async function processSpokeScene( + sceneUrl: string, + projectDir: string, + override: boolean +) { + const url = new URL(sceneUrl); + const fileName = url.pathname.split("/").pop(); + if (fileName) { + await downloadFile({ + url: sceneUrl, + outPath: path.join(projectDir, fileName), + override, + }); + try { + const fileText = readFileSync(path.join(projectDir, fileName), { + encoding: "utf-8", + }) as never; + const parsed = JSON.parse(fileText); + if ("entities" in parsed) { + for (const entity in parsed["entities"]) { + const thisEntity = parsed["entities"][entity]; + if ("components" in thisEntity) { + for (const component of thisEntity["components"]) { + if ("src" in component["props"]) { + const src = component["props"]["src"]; + const assetUrl = new URL(src); + const assetFileName = assetUrl.pathname.split("/").pop(); + await downloadFile({ + url: component["props"]["src"], + outPath: path.join(projectDir, assetFileName), + override, + }); + } + } + } + } + } + } catch (e) { + log.error(e); + } + } +} + async function backupScene( api: ReticulumApi, scene: SceneT, projectDir: string, override: boolean ) { - writeFileSync( - path.join(projectDir, `${scene.scene_id}.json`), - JSON.stringify(scene, null, 2), - { encoding: "utf-8" } - ); + const spokeFilePath = path.join(projectDir, `${scene.scene_id}.json`); + writeFileSync(spokeFilePath, JSON.stringify(scene, null, 2), { + encoding: "utf-8", + }); + + if (scene.scene_project_url) { + await processSpokeScene(scene.scene_project_url, projectDir, override); + } + const url = new URL(scene.model_url); const fileName = url.pathname.split("/").pop(); if (fileName) { @@ -253,6 +300,7 @@ async function backupSpokeProjects( path.join(projectDir, fileName), path.join(projectDir, `${project.name}.spoke`) ); + await processSpokeScene(project.project_url, projectDir, override); } } if (project.thumbnail_url) { diff --git a/src/api/reticulum-api.ts b/src/api/reticulum-api.ts index d340f9a..2c6f669 100644 --- a/src/api/reticulum-api.ts +++ b/src/api/reticulum-api.ts @@ -40,6 +40,7 @@ export interface SceneT { screenshot_url: string; type: ReticulumAssetType; url: string; + scene_project_url?: string; } export interface ProjectT {