-
Notifications
You must be signed in to change notification settings - Fork 1
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add collection management scripts for MOH TOSP #955
Merged
Merged
Changes from 5 commits
Commits
Show all changes
8 commits
Select commit
Hold shift + click to select a range
eb91832
tosp migration scripts
harishv7 e75cfda
add file logger for persistent logs
harishv7 ccbc611
fix lints
harishv7 539a9c1
fix lints
harishv7 03319e7
script enhancements
harishv7 c45f52c
fix filename for log
harishv7 4c47bbe
use generated enums
harishv7 8cb1bb3
use structured inputs and add README
harishv7 File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,46 @@ | ||
import fs from "fs" | ||
import path from "path" | ||
|
||
export class FileLogger { | ||
private logFilePath: string | ||
|
||
constructor(logFilePath: string) { | ||
this.logFilePath = logFilePath | ||
|
||
// Ensure the directory for the log file exists | ||
const logDir = path.dirname(logFilePath) | ||
if (!fs.existsSync(logDir)) { | ||
fs.mkdirSync(logDir, { recursive: true }) | ||
} | ||
} | ||
|
||
private formatLog(level: string, message: string): string { | ||
const timestamp = new Date().toISOString() | ||
return `[${timestamp}] [${level.toUpperCase()}] ${message}\n` | ||
} | ||
|
||
private writeLog(logMessage: string): void { | ||
fs.appendFile(this.logFilePath, logMessage, (err) => { | ||
if (err) { | ||
console.error("Failed to write log:", err) | ||
} | ||
}) | ||
} | ||
|
||
log(level: string, message: string): void { | ||
const logMessage = this.formatLog(level, message) | ||
this.writeLog(logMessage) | ||
} | ||
|
||
info(message: string): void { | ||
this.log("info", message) | ||
} | ||
|
||
error(message: string): void { | ||
this.log("error", message) | ||
} | ||
|
||
debug(message: string): void { | ||
this.log("debug", message) | ||
} | ||
} |
82 changes: 82 additions & 0 deletions
82
apps/studio/prisma/scripts/moh-tosp/backupCollectionById.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,82 @@ | ||
import fs from "fs/promises" // Use the promise-based version of fs for async/await | ||
import path from "path" | ||
|
||
import { db } from "~/server/modules/database" | ||
import { FileLogger } from "../FileLogger" | ||
|
||
// Update the logger path if required | ||
const logger = new FileLogger("./backupCollectionById.log") | ||
|
||
/** | ||
* Backup a collection and its relevant resources to JSON files. | ||
* @param {string} resourceId - ID of the collection resource to back up. | ||
* @param {string} backupDir - Directory to save the backup files. | ||
*/ | ||
export async function backupCollection( | ||
resourceId: string, | ||
backupDir: string, | ||
): Promise<void> { | ||
try { | ||
// Ensure the backup directory exists | ||
await fs.mkdir(backupDir, { recursive: true }) | ||
|
||
// Fetch the collection resource | ||
const collection = await db | ||
.selectFrom("Resource") | ||
.selectAll() | ||
.where("id", "=", resourceId) | ||
.executeTakeFirst() | ||
|
||
if (!collection) { | ||
throw new Error(`Collection with ID ${resourceId} not found.`) | ||
} | ||
|
||
// Fetch all child resources | ||
const children = await db | ||
.selectFrom("Resource") | ||
.selectAll() | ||
.where("parentId", "=", resourceId) | ||
.execute() | ||
|
||
// Write all the children's published version to the backup directory as JSON files | ||
for (const child of children) { | ||
// fetch the blob | ||
const blob = await db | ||
.selectFrom("Blob") | ||
.select("content") | ||
.innerJoin("Version", "Blob.id", "Version.blobId") | ||
.where("Version.id", "=", child.publishedVersionId) | ||
.executeTakeFirst() | ||
|
||
if (!blob) { | ||
throw new Error( | ||
`Published version of child with ID ${child.id} not found.`, | ||
) | ||
} | ||
|
||
logger.info(`Writing backup for child with ID ${child.id}`) | ||
|
||
// Parse blob content and write to a file | ||
const blobBuffer = blob.content // Assuming blob.content is a buffer | ||
const blobJsonPath = path.join(backupDir, `${child.permalink}.json`) | ||
await fs.writeFile(blobJsonPath, JSON.stringify(blobBuffer, null, 2)) | ||
} | ||
|
||
logger.info(`Backup completed successfully in directory: ${backupDir}`) | ||
} catch (error) { | ||
if (error instanceof Error) { | ||
logger.error(`Error backing up collection: ${error.message}`) | ||
} | ||
} | ||
} | ||
|
||
// Run the backup | ||
// NOTE: TODO: Put in the collection ID to backup | ||
const collectionId = "0" | ||
const backupDirectory = "/Users/XYZ/<your-path>" | ||
|
||
await backupCollection(collectionId, backupDirectory).catch((err) => { | ||
if (err instanceof Error) { | ||
logger.error(`Unhandled error: ${err.message}`) | ||
} | ||
}) |
155 changes: 155 additions & 0 deletions
155
apps/studio/prisma/scripts/moh-tosp/createCollectionFromLocal.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,155 @@ | ||
import fs from "fs/promises" | ||
import path from "path" | ||
|
||
import { db, jsonb } from "~/server/modules/database" | ||
import { FileLogger } from "../FileLogger" | ||
|
||
// Update the logger path if required | ||
const logger = new FileLogger("./createCollectionFromLocal.log") | ||
|
||
export const createCollectionFromLocal = async ( | ||
contentDir: string, | ||
siteId: number, | ||
indexPageName: string, // should be placed outside the folder | ||
indexPageTitle: string, // title of the index page | ||
collectionName: string, | ||
nameOfNewCollectionToCreate: string, | ||
) => { | ||
logger.info(`Reading from ${contentDir}`) | ||
const jsonFilePath = path.join(contentDir, indexPageName) | ||
const folderPath = path.join(contentDir, collectionName) | ||
|
||
try { | ||
await db.transaction().execute(async (tx) => { | ||
// Step 1: Create a new collection with title "cost-financing-new" | ||
const collection = await tx | ||
.insertInto("Resource") | ||
.values({ | ||
title: nameOfNewCollectionToCreate, | ||
permalink: nameOfNewCollectionToCreate, | ||
siteId: siteId, | ||
type: "Collection", | ||
state: "Draft", | ||
createdAt: new Date(), | ||
updatedAt: new Date(), | ||
}) | ||
.returning("id") | ||
.executeTakeFirstOrThrow() | ||
|
||
const collectionId = collection.id | ||
logger.info(`Collection created with ID: ${collectionId}`) | ||
|
||
// Step 2: Insert "cost-financing.json" as an IndexPage with permalink "_index" | ||
const jsonFileContent = await fs.readFile(jsonFilePath, "utf-8") | ||
const indexPageBlob = await tx | ||
.insertInto("Blob") | ||
.values({ | ||
content: jsonb(JSON.parse(jsonFileContent)), | ||
}) | ||
.returning("id") | ||
.executeTakeFirstOrThrow() | ||
|
||
const indexPage = await tx | ||
.insertInto("Resource") | ||
.values({ | ||
title: nameOfNewCollectionToCreate, | ||
permalink: "_index", | ||
siteId: siteId, | ||
type: "IndexPage", | ||
parentId: collectionId, | ||
draftBlobId: indexPageBlob.id, | ||
state: "Draft", | ||
createdAt: new Date(), | ||
updatedAt: new Date(), | ||
}) | ||
.returning("id") | ||
.executeTakeFirstOrThrow() | ||
|
||
const indexPageId = indexPage.id | ||
|
||
logger.info(`Index page created with ID: ${indexPageId}`) | ||
|
||
// Step 3: Insert files from "cost-financing/" into the DB as Blobs | ||
const folderFiles = await fs.readdir(folderPath) | ||
logger.info(`Reading from folderPath: ${folderPath}`) | ||
logger.info(`Folder files: ${JSON.stringify(folderFiles)}`) | ||
for (const file of folderFiles) { | ||
const filePath = path.join(folderPath, file) | ||
logger.info(`Reading file path: ${filePath}`) | ||
|
||
logger.info(`Filename: ${file}`) | ||
//Sometimes might have hidden internal files like .DSStore | ||
if (!file.endsWith(".json")) { | ||
continue | ||
} | ||
const fileContent = await fs.readFile(filePath, "utf-8") | ||
|
||
// eslint-disable-next-line @typescript-eslint/no-explicit-any | ||
let parsedFileContent: any | ||
try { | ||
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment | ||
parsedFileContent = JSON.parse(fileContent) | ||
} catch (error) { | ||
if (error instanceof Error) { | ||
logger.error(`Error parsing JSON file: ${file}`) | ||
} | ||
} | ||
|
||
const blob = await tx | ||
.insertInto("Blob") | ||
.values({ | ||
content: parsedFileContent as PrismaJson.BlobJsonContent, | ||
createdAt: new Date(), | ||
updatedAt: new Date(), | ||
}) | ||
.returning("id") | ||
.executeTakeFirstOrThrow() | ||
|
||
const resource = await tx | ||
.insertInto("Resource") | ||
.values({ | ||
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-member-access | ||
title: parsedFileContent.page.title, | ||
permalink: file.replace(/\.json$/, ""), // remove the .json at the back on permalinks | ||
siteId: siteId, // Replace with appropriate site ID | ||
type: "CollectionPage", | ||
parentId: collectionId, | ||
state: "Draft", | ||
draftBlobId: blob.id, | ||
createdAt: new Date(), | ||
updatedAt: new Date(), | ||
}) | ||
.returning("id") | ||
.executeTakeFirstOrThrow() | ||
|
||
const resourceId = resource.id | ||
|
||
logger.info( | ||
`Blob created for file ${file} with resource ID: ${resourceId}`, | ||
) | ||
} | ||
}) | ||
|
||
logger.info("All operations completed successfully.") | ||
} catch (error) { | ||
if (error instanceof Error) { | ||
logger.error(`Error during transaction: ${error.message}`) | ||
} | ||
} | ||
} | ||
|
||
// NOTE: TODO: Update the content directory and siteId here before usage! | ||
const contentDir = "/Users/XYZ/<your-path>" | ||
const indexPagePath = "cost-financing.json" | ||
const indexPageTitle = "Cost financing" | ||
const collectionName = "cost-financing" | ||
const nameOfNewCollectionToCreate = "cost-financing-new" // will also be the permalink | ||
const siteId = 0 | ||
await createCollectionFromLocal( | ||
contentDir, | ||
siteId, | ||
indexPagePath, | ||
indexPageTitle, | ||
collectionName, | ||
nameOfNewCollectionToCreate, | ||
) |
107 changes: 107 additions & 0 deletions
107
apps/studio/prisma/scripts/moh-tosp/deleteCollectionById.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,107 @@ | ||
import { db } from "~/server/modules/database" | ||
import { FileLogger } from "../FileLogger" | ||
|
||
// Update the logger path if required | ||
const logger = new FileLogger("./deleteCollectionById.log") | ||
|
||
export const deleteCollectionById = async ( | ||
collectionId: string, | ||
siteId: number, | ||
) => { | ||
try { | ||
await db.transaction().execute(async (tx) => { | ||
// Step 1: Find all child resources of the collection | ||
const childResources = await tx | ||
.selectFrom("Resource") | ||
.select(["id", "state", "draftBlobId", "publishedVersionId"]) | ||
.where("parentId", "=", collectionId) | ||
.where("siteId", "=", siteId) | ||
.execute() | ||
|
||
// Step 2: Handle each child resource | ||
for (const resource of childResources) { | ||
// Delete published version and its blob, if applicable | ||
if (resource.publishedVersionId) { | ||
const publishedVersion = await tx | ||
.selectFrom("Version") | ||
.select(["blobId"]) | ||
.where("id", "=", resource.publishedVersionId) | ||
.executeTakeFirst() | ||
|
||
const blobIdToDelete = publishedVersion?.blobId | ||
|
||
await tx | ||
.deleteFrom("Version") | ||
.where("id", "=", resource.publishedVersionId) | ||
.execute() | ||
|
||
if (blobIdToDelete) { | ||
await tx | ||
.deleteFrom("Blob") | ||
.where("id", "=", blobIdToDelete) | ||
.execute() | ||
} | ||
} | ||
|
||
// Delete draft blob, if applicable | ||
if (resource.draftBlobId) { | ||
await tx | ||
.deleteFrom("Blob") | ||
.where("id", "=", resource.draftBlobId) | ||
.execute() | ||
} | ||
|
||
// Delete the resource itself | ||
await tx.deleteFrom("Resource").where("id", "=", resource.id).execute() | ||
|
||
logger.info(`Resource with ID ${resource.id} deleted successfully.`) | ||
} | ||
|
||
// Step 3: Delete the collection itself | ||
const collection = await tx | ||
.selectFrom("Resource") | ||
.select(["draftBlobId", "publishedVersionId"]) | ||
.where("id", "=", collectionId) | ||
.executeTakeFirst() | ||
|
||
if (!collection) { | ||
throw new Error(`Collection with ID ${collectionId} not found.`) | ||
} | ||
|
||
// Handle published version and its blob for the collection | ||
if (collection.publishedVersionId) { | ||
const publishedVersion = await tx | ||
.selectFrom("Version") | ||
.select(["blobId"]) | ||
.where("id", "=", collection.publishedVersionId) | ||
.executeTakeFirst() | ||
|
||
const blobIdToDelete = publishedVersion?.blobId | ||
|
||
await tx | ||
.deleteFrom("Version") | ||
.where("id", "=", collection.publishedVersionId) | ||
.execute() | ||
|
||
if (blobIdToDelete) { | ||
await tx.deleteFrom("Blob").where("id", "=", blobIdToDelete).execute() | ||
} | ||
} | ||
|
||
// Delete the collection resource itself | ||
await tx.deleteFrom("Resource").where("id", "=", collectionId).execute() | ||
|
||
logger.info( | ||
`Collection with ID ${collectionId} and all related data deleted successfully.`, | ||
) | ||
}) | ||
} catch (error) { | ||
if (error instanceof Error) { | ||
logger.error(`Error deleting collection: ${error.message}`) | ||
} | ||
} | ||
} | ||
|
||
const collectionIdToDelete = "0" | ||
const siteId = 0 | ||
await deleteCollectionById(collectionIdToDelete, siteId) |
Oops, something went wrong.
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
not sure if i understand why this should be placed outside the folder?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This follows the same structure as what github based repos have due to filesystem based routing, the index page is outside the children folder
e.g. /hello-world.json -> index page
/hello-world/ -> folder