diff --git a/src/components/content-tab-import-export.tsx b/src/components/content-tab-import-export.tsx index 910590bc9..665bed2ea 100644 --- a/src/components/content-tab-import-export.tsx +++ b/src/components/content-tab-import-export.tsx @@ -4,7 +4,7 @@ import { sprintf, __ } from '@wordpress/i18n'; import { Icon, download } from '@wordpress/icons'; import { useI18n } from '@wordpress/react-i18n'; import { useRef } from 'react'; -import { STUDIO_DOCS_URL_IMPORT_EXPORT } from '../constants'; +import { ACCEPTED_IMPORT_FILE_TYPES, STUDIO_DOCS_URL_IMPORT_EXPORT } from '../constants'; import { useConfirmationDialog } from '../hooks/use-confirmation-dialog'; import { useDragAndDropFile } from '../hooks/use-drag-and-drop-file'; import { useImportExport } from '../hooks/use-import-export'; @@ -233,7 +233,7 @@ const ImportSite = ( props: { selectedSite: SiteDetails } ) => { className="hidden" type="file" data-testid="backup-file" - accept=".zip,.sql,.tar,.gz" + accept={ `${ ACCEPTED_IMPORT_FILE_TYPES.join( ',' ) },.sql` } onChange={ onFileSelected } /> diff --git a/src/components/site-form.tsx b/src/components/site-form.tsx index 644ddd6d9..aad06e0ef 100644 --- a/src/components/site-form.tsx +++ b/src/components/site-form.tsx @@ -4,7 +4,7 @@ import { __ } from '@wordpress/i18n'; import { tip, warning, trash, chevronRight, chevronDown, chevronLeft } from '@wordpress/icons'; import { useI18n } from '@wordpress/react-i18n'; import { FormEvent, useRef, useState } from 'react'; -import { STUDIO_DOCS_URL_IMPORT_EXPORT } from '../constants'; +import { ACCEPTED_IMPORT_FILE_TYPES, STUDIO_DOCS_URL_IMPORT_EXPORT } from '../constants'; import { cx } from '../lib/cx'; import { getIpcApi } from '../lib/get-ipc-api'; import Button from './button'; @@ -192,7 +192,7 @@ function FormImportComponent( { className="hidden" type="file" data-testid="backup-file" - accept=".zip,.tar,.gz" + accept={ ACCEPTED_IMPORT_FILE_TYPES.join( ',' ) } onChange={ handleFileChange } /> diff --git a/src/constants.ts b/src/constants.ts index bca455a6b..a286af4dc 100644 --- a/src/constants.ts +++ b/src/constants.ts @@ -25,7 +25,7 @@ export const CHAT_MESSAGES_STORE_KEY = 'ai_chat_messages'; //Import file constants -export const ACCEPTED_IMPORT_FILE_TYPES = [ '.zip', '.gz', '.gzip', '.tar', '.tar.gz' ]; +export const ACCEPTED_IMPORT_FILE_TYPES = [ '.zip', '.gz', '.gzip', '.tar', '.tar.gz', '.wpress' ]; // OAuth constants export const CLIENT_ID = '95109'; diff --git a/src/hooks/tests/use-import-export.test.tsx b/src/hooks/tests/use-import-export.test.tsx index 1fddb3c36..780441695 100644 --- a/src/hooks/tests/use-import-export.test.tsx +++ b/src/hooks/tests/use-import-export.test.tsx @@ -239,7 +239,7 @@ describe( 'useImportExport hook', () => { expect( getIpcApi().showErrorMessageBox ).toHaveBeenCalledWith( { title: 'Failed importing site', message: - 'An error occurred while importing the site. Verify the file is a valid Jetpack backup, Local, Playground or .sql database file and try again. If this problem persists, please contact support.', + 'An error occurred while importing the site. Verify the file is a valid Jetpack backup, Local, Playground, .wpress or .sql database file and try again. If this problem persists, please contact support.', error: 'error', } ); } ); diff --git a/src/hooks/use-import-export.tsx b/src/hooks/use-import-export.tsx index 78acc990d..131faf586 100644 --- a/src/hooks/use-import-export.tsx +++ b/src/hooks/use-import-export.tsx @@ -94,7 +94,7 @@ export const ImportExportProvider = ( { children }: { children: React.ReactNode await getIpcApi().showErrorMessageBox( { title: __( 'Failed importing site' ), message: __( - 'An error occurred while importing the site. Verify the file is a valid Jetpack backup, Local, Playground or .sql database file and try again. If this problem persists, please contact support.' + 'An error occurred while importing the site. Verify the file is a valid Jetpack backup, Local, Playground, .wpress or .sql database file and try again. If this problem persists, please contact support.' ), error, } ); diff --git a/src/lib/import-export/import/handlers/backup-handler-factory.ts b/src/lib/import-export/import/handlers/backup-handler-factory.ts index 420cf3868..af3902a6f 100644 --- a/src/lib/import-export/import/handlers/backup-handler-factory.ts +++ b/src/lib/import-export/import/handlers/backup-handler-factory.ts @@ -2,8 +2,8 @@ import { EventEmitter } from 'events'; import { BackupArchiveInfo } from '../types'; import { BackupHandlerSql } from './backup-handler-sql'; import { BackupHandlerTarGz } from './backup-handler-tar-gz'; +import { BackupHandlerWpress } from './backup-handler-wpress'; import { BackupHandlerZip } from './backup-handler-zip'; - export interface BackupHandler extends Partial< EventEmitter > { listFiles( file: BackupArchiveInfo ): Promise< string[] >; extractFiles( file: BackupArchiveInfo, extractionDirectory: string ): Promise< void >; @@ -54,6 +54,8 @@ export class BackupHandlerFactory { return new BackupHandlerTarGz(); } else if ( this.isSql( file ) ) { return new BackupHandlerSql(); + } else if ( this.isWpress( file ) ) { + return new BackupHandlerWpress(); } } @@ -77,4 +79,8 @@ export class BackupHandlerFactory { this.sqlExtensions.some( ( ext ) => file.path.endsWith( ext ) ) ); } + + private static isWpress( file: BackupArchiveInfo ): boolean { + return file.path.endsWith( '.wpress' ); + } } diff --git a/src/lib/import-export/import/handlers/backup-handler-wpress.ts b/src/lib/import-export/import/handlers/backup-handler-wpress.ts new file mode 100644 index 000000000..1287b08ba --- /dev/null +++ b/src/lib/import-export/import/handlers/backup-handler-wpress.ts @@ -0,0 +1,196 @@ +import { EventEmitter } from 'events'; +import * as fs from 'fs'; +import { constants } from 'fs'; +import * as path from 'path'; +import * as fse from 'fs-extra'; +import { BackupArchiveInfo } from '../types'; +import { BackupHandler } from './backup-handler-factory'; + +/** + * The .wpress format is a custom archive format used by All-In-One WP Migration. + * It is designed to encapsulate all necessary components of a WordPress site, including the database, + * plugins, themes, uploads, and other wp-content files, into a single file for easy transport and restoration. + * + * The .wpress file is structured as follows: + * 1. Header: Contains metadata about the file, such as the name, size, modification time, and prefix. + * The header is a fixed size of 4377 bytes. + * 2. Data Blocks: The actual content of the files, stored in 512-byte blocks. Each file's data is stored + * sequentially, following its corresponding header. + * 3. End of File Marker: A special marker indicating the end of the archive. This is represented by a + * block of 4377 bytes filled with zeroes. + * + * The .wpress format ensures that all necessary components of a WordPress site are included in the backup, + * making it easy to restore the site to its original state. The format is designed to be efficient and + * easy to parse, allowing for quick extraction and restoration of the site's contents. + */ + +const HEADER_SIZE = 4377; +const HEADER_CHUNK_EOF = Buffer.alloc( HEADER_SIZE ); +const CHUNK_SIZE_TO_READ = 1024; + +interface Header { + name: string; + size: number; + mTime: string; + prefix: string; +} + +/** + * Reads a string from a buffer at a given start and end position. + * + * @param {Buffer} buffer - The buffer to read from. + * @param {number} start - The start position of the string in the buffer. + * @param {number} end - The end position of the string in the buffer. + * @returns {string} - The substring buffer, stopping at a null-terminator if present. + */ +function readFromBuffer( buffer: Buffer, start: number, end: number ): string { + const _buffer = buffer.subarray( start, end ); + return _buffer.subarray( 0, _buffer.indexOf( 0x00 ) ).toString(); +} + +/** + * Reads the header of a .wpress file. + * + * @param {fs.promises.FileHandle} fd - The file handle to read from. + * @returns {Promise
} - A promise that resolves to the header or null if the end of the file is reached. + */ +async function readHeader( fd: fs.promises.FileHandle ): Promise< Header | null > { + const headerChunk = Buffer.alloc( HEADER_SIZE ); + await fd.read( headerChunk, 0, HEADER_SIZE ); + + if ( Buffer.compare( headerChunk, HEADER_CHUNK_EOF ) === 0 ) { + return null; + } + + const name = readFromBuffer( headerChunk, 0, 255 ); + const size = parseInt( readFromBuffer( headerChunk, 255, 269 ), 10 ); + const mTime = readFromBuffer( headerChunk, 269, 281 ); + const prefix = readFromBuffer( headerChunk, 281, HEADER_SIZE ); + + return { + name, + size, + mTime, + prefix, + }; +} + +/** + * Reads a block of data from a .wpress file and writes it to a file. + * + * @param {fs.promises.FileHandle} fd - The file handle to read from. + * @param {Header} header - The header of the file to read. + * @param {string} outputPath - The path to write the file to. + */ +async function readBlockToFile( fd: fs.promises.FileHandle, header: Header, outputPath: string ) { + const outputFilePath = path.join( outputPath, header.prefix, header.name ); + await fse.ensureDir( path.dirname( outputFilePath ) ); + const outputStream = fs.createWriteStream( outputFilePath ); + + let totalBytesToRead = header.size; + while ( totalBytesToRead > 0 ) { + let bytesToRead = CHUNK_SIZE_TO_READ; + if ( bytesToRead > totalBytesToRead ) { + bytesToRead = totalBytesToRead; + } + + if ( bytesToRead === 0 ) { + break; + } + + const buffer = Buffer.alloc( bytesToRead ); + const data = await fd.read( buffer, 0, bytesToRead ); + outputStream.write( buffer ); + + totalBytesToRead -= data.bytesRead; + } + + outputStream.close(); +} + +export class BackupHandlerWpress extends EventEmitter implements BackupHandler { + private bytesRead: number; + private eof: Buffer; + + constructor() { + super(); + this.bytesRead = 0; + this.eof = Buffer.alloc( HEADER_SIZE, '\0' ); + } + + /** + * Lists all files in a .wpress backup file by reading the headers sequentially. + * + * It opens the .wpress file, reads each header to get the file names, and stores them in an array. + * The function continues reading headers until it reaches the end of the file. + * + * @param {BackupArchiveInfo} file - The backup archive information, including the file path. + * @returns {Promise} - A promise that resolves to an array of file names. + */ + async listFiles( file: BackupArchiveInfo ): Promise< string[] > { + const fileNames: string[] = []; + + try { + await fs.promises.access( file.path, constants.F_OK ); + } catch ( error ) { + throw new Error( `Input file at location "${ file.path }" could not be found.` ); + } + + const inputFile = await fs.promises.open( file.path, 'r' ); + + // Read all of the headers and file data into memory. + try { + let header; + do { + header = await readHeader( inputFile ); + if ( header ) { + fileNames.push( path.join( header.prefix, header.name ) ); + await inputFile.read( Buffer.alloc( header.size ), 0, header.size, null ); + } + } while ( header ); + } finally { + await inputFile.close(); + } + + return fileNames; + } + + /** + * Extracts files from a .wpress backup file into a specified extraction directory. + * + * @param {BackupArchiveInfo} file - The backup archive information, including the file path. + * @param {string} extractionDirectory - The directory where the files will be extracted. + * @returns {Promise} - A promise that resolves when the extraction is complete. + */ + async extractFiles( file: BackupArchiveInfo, extractionDirectory: string ): Promise< void > { + return new Promise( ( resolve, reject ) => { + ( async () => { + try { + try { + await fs.promises.access( file.path, constants.F_OK ); + } catch ( error ) { + throw new Error( `Input file at location "${ file.path }" could not be found.` ); + } + + await fse.emptyDir( extractionDirectory ); + + const inputFile = await fs.promises.open( file.path, 'r' ); + + let header; + while ( ( header = await readHeader( inputFile ) ) !== null ) { + if ( ! header ) { + break; + } + + await readBlockToFile( inputFile, header, extractionDirectory ); + } + + await inputFile.close(); + resolve(); + } catch ( err ) { + reject( err ); + } + } )(); + } ); + } +} diff --git a/src/lib/import-export/import/import-manager.ts b/src/lib/import-export/import/import-manager.ts index 92c1c7f77..f17c4f125 100644 --- a/src/lib/import-export/import/import-manager.ts +++ b/src/lib/import-export/import/import-manager.ts @@ -11,9 +11,16 @@ import { LocalImporter, PlaygroundImporter, SQLImporter, + WpressImporter, } from './importers/importer'; import { BackupArchiveInfo, NewImporter } from './types'; -import { JetpackValidator, SqlValidator, LocalValidator, PlaygroundValidator } from './validators'; +import { + JetpackValidator, + SqlValidator, + LocalValidator, + PlaygroundValidator, + WpressValidator, +} from './validators'; import { Validator } from './validators/validator'; export interface ImporterOption { @@ -75,4 +82,5 @@ export const defaultImporterOptions: ImporterOption[] = [ { validator: new LocalValidator(), importer: LocalImporter }, { validator: new SqlValidator(), importer: SQLImporter }, { validator: new PlaygroundValidator(), importer: PlaygroundImporter }, + { validator: new WpressValidator(), importer: WpressImporter }, ]; diff --git a/src/lib/import-export/import/importers/importer.ts b/src/lib/import-export/import/importers/importer.ts index 927d3714a..84e0a71af 100644 --- a/src/lib/import-export/import/importers/importer.ts +++ b/src/lib/import-export/import/importers/importer.ts @@ -1,12 +1,14 @@ import { shell } from 'electron'; import { EventEmitter } from 'events'; -import fs from 'fs'; +import fs, { createReadStream, createWriteStream } from 'fs'; import fsPromises from 'fs/promises'; import path from 'path'; +import { createInterface } from 'readline'; import { lstat, move } from 'fs-extra'; import semver from 'semver'; import { DEFAULT_PHP_VERSION } from '../../../../../vendor/wp-now/src/constants'; import { SiteServer } from '../../../../site-server'; +import { serializePlugins } from '../../../serialize-plugins'; import { generateBackupFilename } from '../../export/generate-backup-filename'; import { ImportEvents } from '../events'; import { BackupContents, MetaFileData } from '../types'; @@ -20,6 +22,8 @@ export interface Importer extends Partial< EventEmitter > { } abstract class BaseImporter extends EventEmitter implements Importer { + protected meta?: MetaFileData; + constructor( protected backup: BackupContents ) { super(); } @@ -49,6 +53,7 @@ abstract class BaseImporter extends EventEmitter implements Importer { try { await move( sqlFile, tmpPath ); + await this.prepareSqlFile( tmpPath ); const { stderr, exitCode } = await server.executeWpCliCommand( `sqlite import ${ sqlTempFile } --require=/tmp/sqlite-command/command.php`, // SQLite plugin requires PHP 8+ @@ -71,6 +76,10 @@ abstract class BaseImporter extends EventEmitter implements Importer { this.emit( ImportEvents.IMPORT_DATABASE_COMPLETE ); } + protected async prepareSqlFile( _tmpPath: string ): Promise< void > { + // This method can be overridden by subclasses to prepare the SQL file before import. + } + protected async replaceSiteUrl( siteId: string ) { const server = SiteServer.get( siteId ); if ( ! server ) { @@ -119,17 +128,15 @@ abstract class BaseBackupImporter extends BaseImporter { try { const databaseDir = path.join( rootPath, 'wp-content', 'database' ); const dbPath = path.join( databaseDir, '.ht.sqlite' ); - await this.moveExistingDatabaseToTrash( dbPath ); await this.moveExistingWpContentToTrash( rootPath ); await this.createEmptyDatabase( dbPath ); await this.importWpConfig( rootPath ); await this.importWpContent( rootPath ); - await this.importDatabase( rootPath, siteId, this.backup.sqlFiles ); - let meta: MetaFileData | undefined; if ( this.backup.metaFile ) { - meta = await this.parseMetaFile(); + this.meta = await this.parseMetaFile(); } + await this.importDatabase( rootPath, siteId, this.backup.sqlFiles ); this.emit( ImportEvents.IMPORT_COMPLETE ); return { @@ -138,7 +145,7 @@ abstract class BaseBackupImporter extends BaseImporter { wpContent: this.backup.wpContent, wpContentDirectory: this.backup.wpContentDirectory, wpConfig: this.backup.wpConfig, - meta, + meta: this.meta, }; } catch ( error ) { this.emit( ImportEvents.IMPORT_ERROR, error ); @@ -310,3 +317,100 @@ export class SQLImporter extends BaseImporter { } } } + +export class WpressImporter extends BaseBackupImporter { + protected async parseMetaFile(): Promise< MetaFileData > { + const packageJsonPath = path.join( this.backup.extractionDirectory, 'package.json' ); + try { + const packageContent = await fsPromises.readFile( packageJsonPath, 'utf8' ); + const { + Template: template = '', + Stylesheet: stylesheet = '', + Plugins: plugins = [], + } = JSON.parse( packageContent ); + return { template, stylesheet, plugins }; + } catch ( error ) { + console.error( 'Error reading package.json:', error ); + return { template: '', stylesheet: '', plugins: [] }; + } + } + + protected async prepareSqlFile( tmpPath: string ): Promise< void > { + const tempOutputPath = `${ tmpPath }.tmp`; + const readStream = createReadStream( tmpPath, 'utf8' ); + const writeStream = createWriteStream( tempOutputPath, 'utf8' ); + + const rl = createInterface( { + input: readStream, + crlfDelay: Infinity, + } ); + + rl.on( 'line', ( line: string ) => { + writeStream.write( line.replace( /SERVMASK_PREFIX/g, 'wp' ) + '\n' ); + } ); + + await new Promise( ( resolve, reject ) => { + rl.on( 'close', resolve ); + rl.on( 'error', reject ); + } ); + + await new Promise( ( resolve, reject ) => { + writeStream.end( resolve ); + writeStream.on( 'error', reject ); + } ); + + await fsPromises.rename( tempOutputPath, tmpPath ); + } + + protected async addSqlToSetTheme( sqlFiles: string[] ): Promise< void > { + const { template, stylesheet } = this.meta || {}; + if ( ! template || ! stylesheet ) { + return; + } + + const themeUpdateSql = ` + UPDATE wp_options SET option_value = '${ template }' WHERE option_name = 'template'; + UPDATE wp_options SET option_value = '${ stylesheet }' WHERE option_name = 'stylesheet'; + `; + const sqliteSetThemePath = path.join( + this.backup.extractionDirectory, + 'studio-wpress-theme.sql' + ); + await fsPromises.writeFile( sqliteSetThemePath, themeUpdateSql ); + sqlFiles.push( sqliteSetThemePath ); + } + + protected async addSqlToActivatePlugins( sqlFiles: string[] ): Promise< void > { + const { plugins = [] } = this.meta || {}; + if ( plugins.length === 0 ) { + return; + } + + const serializedPlugins = serializePlugins( plugins ); + const activatePluginsSql = ` + INSERT INTO wp_options (option_name, option_value, autoload) VALUES ('active_plugins', '${ serializedPlugins }', 'yes') + ON CONFLICT(option_name) DO UPDATE SET option_value = excluded.option_value, autoload = excluded.autoload; + `; + + const sqliteActivatePluginsPath = path.join( + this.backup.extractionDirectory, + 'studio-wpress-activate-plugins.sql' + ); + await fsPromises.writeFile( sqliteActivatePluginsPath, activatePluginsSql ); + sqlFiles.push( sqliteActivatePluginsPath ); + } + + protected async importDatabase( + rootPath: string, + siteId: string, + sqlFiles: string[] + ): Promise< void > { + const server = SiteServer.get( siteId ); + if ( ! server ) { + throw new Error( 'Site not found.' ); + } + await this.addSqlToSetTheme( sqlFiles ); + await this.addSqlToActivatePlugins( sqlFiles ); + await super.importDatabase( rootPath, siteId, sqlFiles ); + } +} diff --git a/src/lib/import-export/import/types.ts b/src/lib/import-export/import/types.ts index 676dd535c..1314f74c6 100644 --- a/src/lib/import-export/import/types.ts +++ b/src/lib/import-export/import/types.ts @@ -1,8 +1,11 @@ import { Importer } from './importers'; export interface MetaFileData { - phpVersion: string; - wordpressVersion: string; + phpVersion?: string; + wordpressVersion?: string; + template?: string; + stylesheet?: string; + plugins?: string[]; } export interface WpContent { diff --git a/src/lib/import-export/import/validators/index.ts b/src/lib/import-export/import/validators/index.ts index 1fe38fde9..d1f3a3d83 100644 --- a/src/lib/import-export/import/validators/index.ts +++ b/src/lib/import-export/import/validators/index.ts @@ -3,3 +3,4 @@ export * from './sql-validator'; export * from './jetpack-validator'; export * from './local-validator'; export * from './playground-validator'; +export * from './wpress-validator'; diff --git a/src/lib/import-export/import/validators/wpress-validator.ts b/src/lib/import-export/import/validators/wpress-validator.ts new file mode 100644 index 000000000..7f99c1131 --- /dev/null +++ b/src/lib/import-export/import/validators/wpress-validator.ts @@ -0,0 +1,57 @@ +import { EventEmitter } from 'events'; +import path from 'path'; +import { ImportEvents } from '../events'; +import { BackupContents } from '../types'; +import { Validator } from './validator'; + +export class WpressValidator extends EventEmitter implements Validator { + canHandle( fileList: string[] ): boolean { + const requiredFiles = [ 'database.sql', 'package.json' ]; + const optionalDirs = [ 'uploads', 'plugins', 'themes' ]; + return ( + requiredFiles.every( ( file ) => fileList.includes( file ) ) && + fileList.some( ( file ) => optionalDirs.some( ( dir ) => file.startsWith( dir + '/' ) ) ) + ); + } + + parseBackupContents( fileList: string[], extractionDirectory: string ): BackupContents { + this.emit( ImportEvents.IMPORT_VALIDATION_START ); + const extractedBackup: BackupContents = { + extractionDirectory, + sqlFiles: [], + wpConfig: '', + wpContent: { + uploads: [], + plugins: [], + themes: [], + }, + wpContentDirectory: '', + }; + /* File rules: + * - Accept .wpress + * - Must include database.sql in the root + * - Support optional directories: uploads, plugins, themes, mu-plugins + * */ + + for ( const file of fileList ) { + const fullPath = path.join( extractionDirectory, file ); + if ( file === 'database.sql' ) { + extractedBackup.sqlFiles.push( fullPath ); + } else if ( file.startsWith( 'uploads/' ) ) { + extractedBackup.wpContent.uploads.push( fullPath ); + } else if ( file.startsWith( 'plugins/' ) ) { + extractedBackup.wpContent.plugins.push( fullPath ); + } else if ( file.startsWith( 'themes/' ) ) { + extractedBackup.wpContent.themes.push( fullPath ); + } else if ( file === 'package.json' ) { + extractedBackup.metaFile = fullPath; + } + } + extractedBackup.sqlFiles.sort( ( a: string, b: string ) => + path.basename( a ).localeCompare( path.basename( b ) ) + ); + + this.emit( ImportEvents.IMPORT_VALIDATION_COMPLETE ); + return extractedBackup; + } +} diff --git a/src/lib/import-export/tests/import/validators/wpress-validator.test.ts b/src/lib/import-export/tests/import/validators/wpress-validator.test.ts new file mode 100644 index 000000000..c9381e4d5 --- /dev/null +++ b/src/lib/import-export/tests/import/validators/wpress-validator.test.ts @@ -0,0 +1,87 @@ +import path from 'path'; +import { ImportEvents } from '../../../import/events'; +import { WpressValidator } from '../../../import/validators/wpress-validator'; + +describe( 'WpressValidator', () => { + let validator: WpressValidator; + + beforeEach( () => { + validator = new WpressValidator(); + } ); + + describe( 'canHandle', () => { + it( 'should return true for valid wpress file structure', () => { + const fileList = [ + 'database.sql', + 'package.json', + 'uploads/image.jpg', + 'plugins/some-plugin/plugin.php', + 'themes/some-theme/style.css', + ]; + expect( validator.canHandle( fileList ) ).toBe( true ); + } ); + + it( 'should return false if database.sql is missing', () => { + const fileList = [ + 'package.json', + 'uploads/image.jpg', + 'plugins/some-plugin/plugin.php', + 'themes/some-theme/style.css', + ]; + expect( validator.canHandle( fileList ) ).toBe( false ); + } ); + + it( 'should return false if package.json is missing', () => { + const fileList = [ + 'database.sql', + 'uploads/image.jpg', + 'plugins/some-plugin/plugin.php', + 'themes/some-theme/style.css', + ]; + expect( validator.canHandle( fileList ) ).toBe( false ); + } ); + + it( 'should return false if no optional directories are present', () => { + const fileList = [ 'database.sql', 'package.json', 'some-other-file.txt' ]; + expect( validator.canHandle( fileList ) ).toBe( false ); + } ); + } ); + + describe( 'parseBackupContents', () => { + const extractionDirectory = '/path/to/extraction'; + const fileList = [ + 'database.sql', + 'uploads/image.jpg', + 'plugins/some-plugin/plugin.php', + 'themes/some-theme/style.css', + 'package.json', + ]; + + it( 'should correctly parse backup contents', () => { + const result = validator.parseBackupContents( fileList, extractionDirectory ); + + expect( result.extractionDirectory ).toBe( extractionDirectory ); + expect( result.sqlFiles ).toEqual( [ path.join( extractionDirectory, 'database.sql' ) ] ); + expect( result.wpContent.uploads ).toEqual( [ + path.join( extractionDirectory, 'uploads/image.jpg' ), + ] ); + expect( result.wpContent.plugins ).toEqual( [ + path.join( extractionDirectory, 'plugins/some-plugin/plugin.php' ), + ] ); + expect( result.wpContent.themes ).toEqual( [ + path.join( extractionDirectory, 'themes/some-theme/style.css' ), + ] ); + expect( result.metaFile ).toBe( path.join( extractionDirectory, 'package.json' ) ); + } ); + + it( 'should emit validation events', () => { + const startSpy = jest.spyOn( validator, 'emit' ); + const completeSpy = jest.spyOn( validator, 'emit' ); + + validator.parseBackupContents( fileList, extractionDirectory ); + + expect( startSpy ).toHaveBeenCalledWith( ImportEvents.IMPORT_VALIDATION_START ); + expect( completeSpy ).toHaveBeenCalledWith( ImportEvents.IMPORT_VALIDATION_COMPLETE ); + } ); + } ); +} ); diff --git a/src/lib/serialize-plugins.ts b/src/lib/serialize-plugins.ts new file mode 100644 index 000000000..bdc266880 --- /dev/null +++ b/src/lib/serialize-plugins.ts @@ -0,0 +1,6 @@ +export function serializePlugins( plugins: string[] ): string { + const serializedArray = plugins + .map( ( plugin, index ) => `i:${ index };s:${ plugin.length }:"${ plugin }";` ) + .join( '' ); + return `a:${ plugins.length }:{${ serializedArray }}`; +} diff --git a/src/lib/tests/serialize-plugins.test.ts b/src/lib/tests/serialize-plugins.test.ts new file mode 100644 index 000000000..653639099 --- /dev/null +++ b/src/lib/tests/serialize-plugins.test.ts @@ -0,0 +1,21 @@ +import { serializePlugins } from '../serialize-plugins'; + +describe( 'serializePlugins', () => { + it( 'should correctly serialize an empty array', () => { + const result = serializePlugins( [] ); + expect( result ).toBe( 'a:0:{}' ); + } ); + + it( 'should correctly serialize an array with one plugin', () => { + const result = serializePlugins( [ 'hello-dolly' ] ); + expect( result ).toBe( 'a:1:{i:0;s:11:"hello-dolly";}' ); + } ); + + it( 'should correctly serialize an array with multiple plugins', () => { + const plugins = [ 'akismet', 'jetpack', 'woocommerce', 'classc-editor' ]; + const result = serializePlugins( plugins ); + expect( result ).toBe( + 'a:4:{i:0;s:7:"akismet";i:1;s:7:"jetpack";i:2;s:11:"woocommerce";i:3;s:13:"classc-editor";}' + ); + } ); +} );