args['type'] ?? ''; $events = $this->args['events'] ?? []; $project = new Document($this->args['project'] ?? []); $user = new Document($this->args['user'] ?? []); $payload = json_encode($this->args['payload'] ?? []); if ($project->getId() === 'console') { return; } /** * Handle Event execution. */ if (! empty($events)) { return; } $this->dbForProject = $this->getProjectDB($project); $this->processMigration(); } /** * Process Source * * @return Source * * @throws \Exception */ protected function processSource(string $source, array $credentials): Source { switch ($source) { case Firebase::getName(): return new Firebase( json_decode($credentials['serviceAccount'], true), ); break; case Supabase::getName(): return new Supabase( $credentials['endpoint'], $credentials['apiKey'], $credentials['databaseHost'], 'postgres', $credentials['username'], $credentials['password'], $credentials['port'], ); break; case NHost::getName(): return new NHost( $credentials['subdomain'], $credentials['region'], $credentials['adminSecret'], $credentials['database'], $credentials['username'], $credentials['password'], $credentials['port'], ); break; case Appwrite::getName(): return new Appwrite($credentials['projectId'], str_starts_with($credentials['endpoint'], 'http://localhost/v1') ? 'http://appwrite/v1' : $credentials['endpoint'], $credentials['apiKey']); break; default: throw new \Exception('Invalid source type'); break; } } protected function updateMigrationDocument(Document $migration, Document $project): Document { /** Trigger Realtime */ $allEvents = Event::generateEvents('migrations.[migrationId].update', [ 'migrationId' => $migration->getId(), ]); $target = Realtime::fromPayload( event: $allEvents[0], payload: $migration, project: $project ); Realtime::send( projectId: 'console', payload: $migration->getArrayCopy(), events: $allEvents, channels: $target['channels'], roles: $target['roles'], ); Realtime::send( projectId: $project->getId(), payload: $migration->getArrayCopy(), events: $allEvents, channels: $target['channels'], roles: $target['roles'], ); return $this->dbForProject->updateDocument('migrations', $migration->getId(), $migration); } protected function removeAPIKey(Document $apiKey) { $consoleDB = $this->getConsoleDB(); $consoleDB->deleteDocument('keys', $apiKey->getId()); } protected function generateAPIKey(Document $project): Document { $consoleDB = $this->getConsoleDB(); $generatedSecret = bin2hex(\random_bytes(128)); $key = new Document([ '$id' => ID::unique(), '$permissions' => [ Permission::read(Role::any()), Permission::update(Role::any()), Permission::delete(Role::any()), ], 'projectInternalId' => $project->getInternalId(), 'projectId' => $project->getId(), 'name' => 'Transfer API Key', 'scopes' => [ 'users.read', 'users.write', 'teams.read', 'teams.write', 'databases.read', 'databases.write', 'collections.read', 'collections.write', 'documents.read', 'documents.write', 'buckets.read', 'buckets.write', 'files.read', 'files.write', 'functions.read', 'functions.write', ], 'expire' => null, 'sdks' => [], 'accessedAt' => null, 'secret' => $generatedSecret, ]); $consoleDB->createDocument('keys', $key); $consoleDB->deleteCachedDocument('projects', $project->getId()); return $key; } /** * Process Migration * * @return void */ protected function processMigration(): void { /** * @var Document $migrationDocument * @var Transfer $transfer */ $migrationDocument = null; $transfer = null; $projectDocument = $this->getConsoleDB()->getDocument('projects', $this->args['project']['$id']); $tempAPIKey = $this->generateAPIKey($projectDocument); try { $migrationDocument = $this->dbForProject->getDocument('migrations', $this->args['migration']['$id']); $migrationDocument->setAttribute('stage', 'processing'); $migrationDocument->setAttribute('status', 'processing'); $this->updateMigrationDocument($migrationDocument, $projectDocument); $source = $this->processSource($migrationDocument->getAttribute('source'), $migrationDocument->getAttribute('credentials')); $source->report(); $destination = new DestinationsAppwrite( $projectDocument->getId(), 'http://appwrite/v1', $tempAPIKey['secret'], ); $transfer = new Transfer( $source, $destination ); /** Start Transfer */ $migrationDocument->setAttribute('stage', 'migrating'); $this->updateMigrationDocument($migrationDocument, $projectDocument); $transfer->run($migrationDocument->getAttribute('resources'), function () use ($migrationDocument, $transfer, $projectDocument) { $migrationDocument->setAttribute('resourceData', json_encode($transfer->getCache())); $migrationDocument->setAttribute('statusCounters', json_encode($transfer->getStatusCounters())); $this->updateMigrationDocument($migrationDocument, $projectDocument); }); $errors = $transfer->getReport(Resource::STATUS_ERROR); if (count($errors) > 0) { $migrationDocument->setAttribute('status', 'failed'); $migrationDocument->setAttribute('stage', 'finished'); $errorMessages = []; foreach ($errors as $error) { $errorMessages[] = "Failed to transfer resource '{$error['id']}:{$error['resource']}' with message '{$error['message']}'"; } $migrationDocument->setAttribute('errors', $errorMessages); $this->updateMigrationDocument($migrationDocument, $projectDocument); return; } $migrationDocument->setAttribute('status', 'completed'); $migrationDocument->setAttribute('stage', 'finished'); } catch (\Throwable $th) { Console::error($th->getMessage()); if ($migrationDocument) { Console::error($th->getMessage()); Console::error($th->getTraceAsString()); $migrationDocument->setAttribute('status', 'failed'); $migrationDocument->setAttribute('stage', 'finished'); $migrationDocument->setAttribute('errors', [$th->getMessage()]); return; } if ($transfer) { $errors = $transfer->getReport(Resource::STATUS_ERROR); if (count($errors) > 0) { $migrationDocument->setAttribute('status', 'failed'); $migrationDocument->setAttribute('stage', 'finished'); $migrationDocument->setAttribute('errors', $errors); } } } finally { if ($migrationDocument) { $this->updateMigrationDocument($migrationDocument, $projectDocument); } if ($tempAPIKey) { $this->removeAPIKey($tempAPIKey); } } } /** * Process Verification * * @return void */ protected function processVerification(): void { } public function shutdown(): void { } }