Some checks failed
ci / changes (push) Has been cancelled
ci / lint (push) Has been cancelled
ci / build (push) Has been cancelled
ci / tests-unit (push) Has been cancelled
ci / tests-types (push) Has been cancelled
ci / int-cosmosdb (push) Has been cancelled
ci / int-documentdb (push) Has been cancelled
ci / int-firestore (push) Has been cancelled
ci / int-mongodb (push) Has been cancelled
ci / int-postgres (push) Has been cancelled
ci / int-postgres-custom-schema (push) Has been cancelled
ci / int-postgres-uuid (push) Has been cancelled
ci / int-sqlite (push) Has been cancelled
ci / int-sqlite-uuid (push) Has been cancelled
ci / int-supabase (push) Has been cancelled
ci / e2e-_community (push) Has been cancelled
ci / e2e-access-control (push) Has been cancelled
ci / e2e-admin-bar (push) Has been cancelled
ci / e2e-admin-root (push) Has been cancelled
ci / e2e-admin__e2e__document-view (push) Has been cancelled
ci / e2e-admin__e2e__general (push) Has been cancelled
ci / e2e-admin__e2e__list-view (push) Has been cancelled
ci / e2e-auth (push) Has been cancelled
ci / e2e-auth-basic (push) Has been cancelled
ci / e2e-bulk-edit (push) Has been cancelled
ci / e2e-field-error-states (push) Has been cancelled
ci / e2e-fields-relationship (push) Has been cancelled
ci / e2e-fields__collections__Array (push) Has been cancelled
ci / e2e-fields__collections__Blocks#config.blockreferences.ts (push) Has been cancelled
ci / e2e-fields__collections__Blocks (push) Has been cancelled
ci / e2e-fields__collections__Checkbox (push) Has been cancelled
ci / e2e-fields__collections__Collapsible (push) Has been cancelled
ci / e2e-fields__collections__ConditionalLogic (push) Has been cancelled
ci / e2e-fields__collections__CustomID (push) Has been cancelled
ci / e2e-fields__collections__Date (push) Has been cancelled
ci / e2e-fields__collections__Email (push) Has been cancelled
ci / e2e-fields__collections__Indexed (push) Has been cancelled
ci / e2e-fields__collections__JSON (push) Has been cancelled
ci / e2e-fields__collections__Number (push) Has been cancelled
ci / e2e-fields__collections__Point (push) Has been cancelled
ci / e2e-fields__collections__Radio (push) Has been cancelled
ci / e2e-fields__collections__Relationship (push) Has been cancelled
ci / e2e-fields__collections__Row (push) Has been cancelled
ci / e2e-fields__collections__Select (push) Has been cancelled
ci / e2e-fields__collections__Tabs (push) Has been cancelled
ci / e2e-fields__collections__Tabs2 (push) Has been cancelled
ci / e2e-fields__collections__Text (push) Has been cancelled
ci / e2e-fields__collections__UI (push) Has been cancelled
ci / e2e-fields__collections__Upload (push) Has been cancelled
ci / e2e-folders (push) Has been cancelled
ci / e2e-form-state (push) Has been cancelled
ci / e2e-group-by (push) Has been cancelled
ci / e2e-hooks (push) Has been cancelled
ci / e2e-i18n (push) Has been cancelled
ci / e2e-joins (push) Has been cancelled
ci / e2e-lexical__collections__LexicalHeadingFeature (push) Has been cancelled
ci / e2e-lexical__collections__LexicalJSXConverter (push) Has been cancelled
ci / e2e-lexical__collections__LexicalLinkFeature (push) Has been cancelled
ci / e2e-lexical__collections__Lexical__e2e__blocks#config.blockreferences.ts (push) Has been cancelled
ci / e2e-lexical__collections__Lexical__e2e__blocks (push) Has been cancelled
ci / e2e-lexical__collections__Lexical__e2e__main (push) Has been cancelled
ci / e2e-lexical__collections__OnDemandForm (push) Has been cancelled
ci / e2e-lexical__collections__RichText (push) Has been cancelled
ci / e2e-lexical__collections___LexicalFullyFeatured (push) Has been cancelled
ci / e2e-lexical__collections___LexicalFullyFeatured__db (push) Has been cancelled
ci / e2e-live-preview (push) Has been cancelled
ci / e2e-localization (push) Has been cancelled
ci / e2e-locked-documents (push) Has been cancelled
ci / e2e-plugin-cloud-storage (push) Has been cancelled
ci / e2e-plugin-form-builder (push) Has been cancelled
ci / e2e-plugin-import-export (push) Has been cancelled
ci / e2e-plugin-multi-tenant (push) Has been cancelled
ci / e2e-plugin-nested-docs (push) Has been cancelled
ci / e2e-plugin-seo (push) Has been cancelled
ci / e2e-query-presets (push) Has been cancelled
ci / e2e-sort (push) Has been cancelled
ci / e2e-trash (push) Has been cancelled
ci / e2e-uploads (push) Has been cancelled
ci / e2e-versions (push) Has been cancelled
ci / e2e-turbo-_community (push) Has been cancelled
ci / e2e-turbo-access-control (push) Has been cancelled
ci / e2e-turbo-admin-bar (push) Has been cancelled
ci / e2e-turbo-admin-root (push) Has been cancelled
ci / e2e-turbo-admin__e2e__document-view (push) Has been cancelled
ci / e2e-turbo-admin__e2e__general (push) Has been cancelled
ci / e2e-turbo-admin__e2e__list-view (push) Has been cancelled
ci / e2e-turbo-auth (push) Has been cancelled
ci / e2e-turbo-auth-basic (push) Has been cancelled
ci / e2e-turbo-bulk-edit (push) Has been cancelled
ci / e2e-turbo-field-error-states (push) Has been cancelled
ci / e2e-turbo-fields-relationship (push) Has been cancelled
ci / e2e-turbo-fields__collections__Array (push) Has been cancelled
ci / e2e-turbo-fields__collections__Blocks#config.blockreferences.ts (push) Has been cancelled
ci / e2e-turbo-fields__collections__Blocks (push) Has been cancelled
ci / e2e-turbo-fields__collections__Checkbox (push) Has been cancelled
ci / e2e-turbo-fields__collections__Collapsible (push) Has been cancelled
ci / e2e-turbo-fields__collections__ConditionalLogic (push) Has been cancelled
ci / e2e-turbo-fields__collections__CustomID (push) Has been cancelled
ci / e2e-turbo-fields__collections__Date (push) Has been cancelled
ci / e2e-turbo-fields__collections__Email (push) Has been cancelled
ci / e2e-turbo-fields__collections__Indexed (push) Has been cancelled
ci / e2e-turbo-fields__collections__JSON (push) Has been cancelled
ci / e2e-turbo-fields__collections__Number (push) Has been cancelled
ci / e2e-turbo-fields__collections__Point (push) Has been cancelled
ci / e2e-turbo-fields__collections__Radio (push) Has been cancelled
ci / e2e-turbo-fields__collections__Relationship (push) Has been cancelled
ci / e2e-turbo-fields__collections__Row (push) Has been cancelled
ci / e2e-turbo-fields__collections__Select (push) Has been cancelled
ci / e2e-turbo-fields__collections__Tabs (push) Has been cancelled
ci / e2e-turbo-fields__collections__Tabs2 (push) Has been cancelled
ci / e2e-turbo-fields__collections__Text (push) Has been cancelled
ci / e2e-turbo-fields__collections__UI (push) Has been cancelled
ci / e2e-turbo-fields__collections__Upload (push) Has been cancelled
ci / e2e-turbo-folders (push) Has been cancelled
ci / e2e-turbo-form-state (push) Has been cancelled
ci / e2e-turbo-group-by (push) Has been cancelled
ci / e2e-turbo-hooks (push) Has been cancelled
ci / e2e-turbo-i18n (push) Has been cancelled
ci / e2e-turbo-joins (push) Has been cancelled
ci / e2e-turbo-lexical__collections__LexicalHeadingFeature (push) Has been cancelled
ci / e2e-turbo-lexical__collections__LexicalJSXConverter (push) Has been cancelled
ci / e2e-turbo-lexical__collections__LexicalLinkFeature (push) Has been cancelled
ci / e2e-turbo-lexical__collections__Lexical__e2e__blocks#config.blockreferences.ts (push) Has been cancelled
ci / e2e-turbo-lexical__collections__Lexical__e2e__blocks (push) Has been cancelled
ci / e2e-turbo-lexical__collections__Lexical__e2e__main (push) Has been cancelled
ci / e2e-turbo-lexical__collections__OnDemandForm (push) Has been cancelled
ci / e2e-turbo-lexical__collections__RichText (push) Has been cancelled
ci / e2e-turbo-lexical__collections___LexicalFullyFeatured (push) Has been cancelled
ci / e2e-turbo-lexical__collections___LexicalFullyFeatured__db (push) Has been cancelled
ci / e2e-turbo-live-preview (push) Has been cancelled
ci / e2e-turbo-localization (push) Has been cancelled
ci / e2e-turbo-locked-documents (push) Has been cancelled
ci / e2e-turbo-plugin-cloud-storage (push) Has been cancelled
ci / e2e-turbo-plugin-form-builder (push) Has been cancelled
ci / e2e-turbo-plugin-import-export (push) Has been cancelled
ci / e2e-turbo-plugin-multi-tenant (push) Has been cancelled
ci / e2e-turbo-plugin-nested-docs (push) Has been cancelled
ci / e2e-turbo-plugin-seo (push) Has been cancelled
ci / e2e-turbo-query-presets (push) Has been cancelled
ci / e2e-turbo-sort (push) Has been cancelled
ci / e2e-turbo-trash (push) Has been cancelled
ci / e2e-turbo-uploads (push) Has been cancelled
ci / e2e-turbo-versions (push) Has been cancelled
ci / build-template-blank-mongodb (push) Has been cancelled
ci / build-template-website-mongodb (push) Has been cancelled
ci / build-template-with-payload-cloud-mongodb (push) Has been cancelled
ci / build-template-with-vercel-mongodb-mongodb (push) Has been cancelled
ci / build-template-plugin- (push) Has been cancelled
ci / build-template-with-postgres-postgres (push) Has been cancelled
ci / build-template-with-vercel-postgres-postgres (push) Has been cancelled
ci / tests-type-generation (push) Has been cancelled
ci / All Green (push) Has been cancelled
ci / Publish Canary (push) Has been cancelled
ci / analyze (push) Has been cancelled
publish-prerelease / publish-prerelease-${{ github.ref_name }}-${{ github.sha }} (push) Has been cancelled
lock-issues / lock_issues (push) Has been cancelled
stale / stale (push) Has been cancelled
audit-dependencies / audit (push) Has been cancelled
activity-notifications / run (push) Has been cancelled
151 lines
4.9 KiB
TypeScript
151 lines
4.9 KiB
TypeScript
import type { PostgresAdapter } from '@tabshiftcms/db-postgres/types'
|
|
import type { SQLiteAdapter } from '@tabshiftcms/db-sqlite/types'
|
|
import type { PgTable } from 'drizzle-orm/pg-core'
|
|
import type { SQLiteTable } from 'drizzle-orm/sqlite-core'
|
|
import type { Payload } from 'tbsh-cms'
|
|
|
|
import { sql } from 'drizzle-orm'
|
|
|
|
import { isMongoose } from './isMongoose.js'
|
|
|
|
export const uploadsDirCache: {
|
|
[key: string]: {
|
|
cacheDir: string
|
|
originalDir: string
|
|
}[]
|
|
} = {}
|
|
export const dbSnapshot = {}
|
|
|
|
async function createMongooseSnapshot(collectionsObj, snapshotKey: string) {
|
|
const snapshot = {}
|
|
|
|
// Assuming `collectionsObj` is an object where keys are names and values are collection references
|
|
for (const collectionName of Object.keys(collectionsObj)) {
|
|
const collection = collectionsObj[collectionName]
|
|
const documents = await collection.find({}).toArray() // Get all documents
|
|
snapshot[collectionName] = documents
|
|
}
|
|
|
|
dbSnapshot[snapshotKey] = snapshot // Save the snapshot in memory
|
|
}
|
|
|
|
async function restoreFromMongooseSnapshot(collectionsObj, snapshotKey: string) {
|
|
if (!dbSnapshot[snapshotKey]) {
|
|
throw new Error('No snapshot found to restore from.')
|
|
}
|
|
|
|
// Assuming `collectionsObj` is an object where keys are names and values are collection references
|
|
for (const [name, documents] of Object.entries(dbSnapshot[snapshotKey])) {
|
|
const collection = collectionsObj[name]
|
|
// You would typically clear the collection here, but as per your requirement, you do it manually
|
|
if ((documents as any[]).length > 0) {
|
|
await collection.insertMany(documents)
|
|
}
|
|
}
|
|
}
|
|
|
|
async function createDrizzleSnapshot(db: PostgresAdapter | SQLiteAdapter, snapshotKey: string) {
|
|
const snapshot = {}
|
|
|
|
const schema: Record<string, PgTable | SQLiteTable> = db.drizzle._.schema
|
|
if (!schema) {
|
|
return
|
|
}
|
|
|
|
for (const tableName in schema) {
|
|
const table = db.drizzle.query[tableName]['fullSchema'][tableName] //db.drizzle._.schema[tableName]
|
|
const records = await db.drizzle.select().from(table).execute()
|
|
snapshot[tableName] = records
|
|
}
|
|
|
|
dbSnapshot[snapshotKey] = snapshot
|
|
}
|
|
|
|
async function restoreFromDrizzleSnapshot(
|
|
adapter: PostgresAdapter | SQLiteAdapter,
|
|
snapshotKey: string,
|
|
) {
|
|
if (!dbSnapshot[snapshotKey]) {
|
|
throw new Error('No snapshot found to restore from.')
|
|
}
|
|
const db = adapter.name === 'postgres' ? (adapter as PostgresAdapter) : (adapter as SQLiteAdapter)
|
|
let disableFKConstraintChecksQuery
|
|
let enableFKConstraintChecksQuery
|
|
|
|
if (db.name === 'sqlite') {
|
|
disableFKConstraintChecksQuery = 'PRAGMA foreign_keys = off'
|
|
enableFKConstraintChecksQuery = 'PRAGMA foreign_keys = on'
|
|
}
|
|
if (db.name === 'postgres') {
|
|
disableFKConstraintChecksQuery = 'SET session_replication_role = replica;'
|
|
enableFKConstraintChecksQuery = 'SET session_replication_role = DEFAULT;'
|
|
}
|
|
|
|
// Temporarily disable foreign key constraint checks
|
|
try {
|
|
await db.execute({
|
|
drizzle: db.drizzle,
|
|
raw: disableFKConstraintChecksQuery,
|
|
})
|
|
for (const tableName in dbSnapshot[snapshotKey]) {
|
|
const table = db.drizzle.query[tableName]['fullSchema'][tableName]
|
|
await db.execute({
|
|
drizzle: db.drizzle,
|
|
sql: sql`DELETE FROM ${table}`,
|
|
}) // This deletes all records from the table. Probably not necessary, as I'm deleting the table before restoring anyways
|
|
|
|
const records = dbSnapshot[snapshotKey][tableName]
|
|
if (records.length > 0) {
|
|
await db.drizzle.insert(table).values(records).execute()
|
|
}
|
|
}
|
|
} catch (e) {
|
|
console.error(e)
|
|
} finally {
|
|
// Re-enable foreign key constraint checks
|
|
await db.execute({
|
|
drizzle: db.drizzle,
|
|
raw: enableFKConstraintChecksQuery,
|
|
})
|
|
}
|
|
}
|
|
|
|
export async function createSnapshot(
|
|
_payload: Payload,
|
|
snapshotKey: string,
|
|
collectionSlugs: string[],
|
|
) {
|
|
if (isMongoose(_payload) && 'collections' in _payload.db) {
|
|
const firstCollectionSlug = collectionSlugs?.[0]
|
|
|
|
if (!firstCollectionSlug?.length) {
|
|
throw new Error('No collection slugs provided to reset the database.')
|
|
}
|
|
|
|
const mongooseCollections = _payload.db.collections[firstCollectionSlug]?.db.collections
|
|
|
|
await createMongooseSnapshot(mongooseCollections, snapshotKey)
|
|
} else {
|
|
const db: PostgresAdapter = _payload.db as unknown as PostgresAdapter
|
|
await createDrizzleSnapshot(db, snapshotKey)
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Make sure to delete the db before calling this function
|
|
* @param _payload
|
|
*/
|
|
export async function restoreFromSnapshot(
|
|
_payload: Payload,
|
|
snapshotKey: string,
|
|
collectionSlugs: string[],
|
|
) {
|
|
if (isMongoose(_payload) && 'collections' in _payload.db) {
|
|
const mongooseCollections = _payload.db.collections[collectionSlugs[0]].db.collections
|
|
await restoreFromMongooseSnapshot(mongooseCollections, snapshotKey)
|
|
} else {
|
|
const db: PostgresAdapter = _payload.db as unknown as PostgresAdapter
|
|
await restoreFromDrizzleSnapshot(db, snapshotKey)
|
|
}
|
|
}
|