2012-02-03 20:32:06 +00:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* ownCloud
|
|
|
|
*
|
|
|
|
* @author Tom Needham
|
|
|
|
* @copyright 2012 Tom Needham tom@owncloud.com
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 3 of the License, or any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU AFFERO GENERAL PUBLIC LICENSE for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public
|
|
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* provides an interface to all search providers
|
|
|
|
*/
|
|
|
|
class OC_Migrate{
|
2012-03-09 23:33:11 +00:00
|
|
|
|
2012-03-13 16:21:17 +00:00
|
|
|
// Holds the db object
|
2012-03-09 23:33:11 +00:00
|
|
|
static private $MDB2=false;
|
2012-03-13 16:21:17 +00:00
|
|
|
// Array of OC_Migration_Provider objects
|
2012-02-03 21:00:12 +00:00
|
|
|
static private $providers=array();
|
2012-03-13 16:21:17 +00:00
|
|
|
// Schema db object
|
2012-03-09 23:33:11 +00:00
|
|
|
static private $schema=false;
|
2012-03-13 16:21:17 +00:00
|
|
|
// User id of the user to import/export
|
2012-03-10 18:18:58 +00:00
|
|
|
static private $uid=false;
|
2012-03-13 16:21:17 +00:00
|
|
|
// Path to the sqlite db
|
|
|
|
static private $dbpath=false;
|
2012-03-13 23:09:43 +00:00
|
|
|
// Holds the ZipArchive object
|
|
|
|
static private $zip=false;
|
|
|
|
// String path to export
|
|
|
|
static private $zippath=false;
|
2012-03-16 21:09:36 +00:00
|
|
|
// Stores the type of export
|
|
|
|
static private $exporttype=false;
|
|
|
|
// Array of temp files to be deleted after zip creation
|
|
|
|
static private $tmpfiles=array();
|
2012-02-03 20:32:06 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* register a new migration provider
|
|
|
|
* @param OC_Migrate_Provider $provider
|
|
|
|
*/
|
|
|
|
public static function registerProvider($provider){
|
|
|
|
self::$providers[]=$provider;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2012-03-09 23:33:11 +00:00
|
|
|
* @breif creates a migration.db in the users data dir with their app data in
|
|
|
|
* @return bool whether operation was successfull
|
2012-02-03 20:32:06 +00:00
|
|
|
*/
|
2012-03-13 23:09:43 +00:00
|
|
|
private static function exportAppData( ){
|
|
|
|
|
2012-03-10 18:18:58 +00:00
|
|
|
self::connectDB();
|
|
|
|
$ok = true;
|
|
|
|
$return = array();
|
|
|
|
|
2012-03-13 16:21:17 +00:00
|
|
|
// Find the providers
|
|
|
|
$apps = OC_App::getAllApps();
|
|
|
|
|
|
|
|
foreach($apps as $app){
|
|
|
|
$path = OC::$SERVERROOT . '/apps/' . $app . '/lib/migrate.php';
|
|
|
|
if( file_exists( $path ) ){
|
|
|
|
include( $path );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-09 23:33:11 +00:00
|
|
|
// Foreach provider
|
2012-03-10 18:18:58 +00:00
|
|
|
foreach( self::$providers as $provider ){
|
2012-03-12 18:40:14 +00:00
|
|
|
$failed = false;
|
|
|
|
|
|
|
|
// Does this app use the database?
|
2012-03-10 18:18:58 +00:00
|
|
|
if(file_exists(OC::$SERVERROOT.'/apps/'.$provider->id.'/appinfo/database.xml')){
|
2012-03-12 18:40:14 +00:00
|
|
|
// Create some app tables
|
|
|
|
$tables = self::createAppTables( $provider->id );
|
|
|
|
if( is_array( $tables ) ){
|
|
|
|
// Save the table names
|
|
|
|
foreach($tables as $table){
|
|
|
|
$return['app'][$provider->id]['tables'][] = $table;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// It failed to create the tables
|
|
|
|
$failed = true;
|
|
|
|
}
|
2012-03-10 18:18:58 +00:00
|
|
|
}
|
2012-03-12 18:40:14 +00:00
|
|
|
|
|
|
|
// Run the import function?
|
|
|
|
if( !$failed ){
|
2012-03-13 23:09:43 +00:00
|
|
|
$return['app'][$provider->id]['success'] = $provider->export( self::$uid );
|
2012-03-11 22:09:16 +00:00
|
|
|
} else {
|
2012-03-12 18:40:14 +00:00
|
|
|
$return['app'][$provider->id]['success'] = false;
|
|
|
|
$return['app'][$provider->id]['message'] = 'failed to create the app tables';
|
2012-03-10 18:18:58 +00:00
|
|
|
}
|
2012-03-12 18:40:14 +00:00
|
|
|
|
|
|
|
// Now add some app info the the return array
|
|
|
|
$appinfo = OC_App::getAppInfo( $provider->id );
|
|
|
|
$return['app'][$provider->id]['version'] = $appinfo['version'];
|
|
|
|
|
2012-03-09 23:33:11 +00:00
|
|
|
}
|
2012-03-12 18:40:14 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Add some general info to the return array
|
2012-03-13 23:09:43 +00:00
|
|
|
$return['migrateinfo']['uid'] = self::$uid;
|
2012-03-12 18:40:14 +00:00
|
|
|
$return['migrateinfo']['ocversion'] = OC_Util::getVersionString();
|
2012-03-09 23:33:11 +00:00
|
|
|
|
2012-03-10 18:18:58 +00:00
|
|
|
return $return;
|
2012-03-09 23:33:11 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-03-15 20:52:43 +00:00
|
|
|
/**
|
|
|
|
* @breif creates an export file for the whole system
|
|
|
|
* @param optional $exporttype string export type ('instance','system' or 'userfiles')
|
|
|
|
* @param optional $path string path to zip destination (with trailing slash)
|
|
|
|
* @return path to the zip or false if there was a problem
|
|
|
|
*/
|
|
|
|
static public function createSysExportFile( $exporttype='instance', $path=null ){
|
|
|
|
// Calculate zip name
|
|
|
|
$zipname = "owncloud_export_" . date("y-m-d_H-i-s") . ".zip";
|
|
|
|
// Get the data dir
|
|
|
|
$datadir = OC_Config::getValue( 'datadirectory' );
|
|
|
|
// Calculate destination
|
|
|
|
if( !is_null( $path ) ){
|
|
|
|
// Path given
|
|
|
|
// Is a directory?
|
|
|
|
if( !is_dir( $path ) ){
|
|
|
|
OC_Log::write('migration', 'Path supplied to createSysExportFile() is not a directory', OC_Log::ERROR);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// Is writeable
|
|
|
|
if( !is_writeable( $path ) ){
|
|
|
|
OC_Log::write('migration', 'Path supplied to createSysExportFile() is not writeable', OC_Log::ERROR);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
self::$zippath = $path . $zipname;
|
|
|
|
} else {
|
|
|
|
// Save in tmp dir
|
2012-03-16 21:09:36 +00:00
|
|
|
self::$zippath = sys_get_temp_dir() . '/' . $zipname;
|
2012-03-15 20:52:43 +00:00
|
|
|
}
|
|
|
|
// Create the zip object
|
|
|
|
self::$zip = new ZipArchive;
|
|
|
|
// Try to create the zip
|
|
|
|
if( !self::createZip() ){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// Handle export types
|
2012-03-16 21:09:36 +00:00
|
|
|
$exporttypes = array( 'userfiles', 'instance', 'system' );
|
|
|
|
self::$exporttype = in_array( $exporttype, $exporttypes ) ? $exporttype : false;
|
|
|
|
if( !self::$exporttype ){
|
|
|
|
OC_Log::write( 'migration', 'Export type: '.$exporttype.' is not supported.', OC_Log::ERROR);
|
2012-03-15 20:52:43 +00:00
|
|
|
return false;
|
|
|
|
}
|
2012-03-16 21:09:36 +00:00
|
|
|
switch( self::$exporttype ){
|
|
|
|
case 'instance':
|
|
|
|
// Creates a zip that is compatable with the import function
|
|
|
|
$dbfile = tempnam( "/tmp", "owncloud_export_data_" );
|
|
|
|
OC_DB::getDbStructure( $dbfile, 'MDB2_SCHEMA_DUMP_ALL');
|
|
|
|
|
|
|
|
// Now add in *dbname* and *dbprefix*
|
|
|
|
$dbexport = file_get_contents( $dbfile );
|
|
|
|
$dbnamestring = "<database>\n\n <name>" . OC_Config::getValue( "dbname", "owncloud" );
|
|
|
|
$dbtableprefixstring = "<table>\n\n <name>" . OC_Config::getValue( "dbtableprefix", "oc_" );
|
|
|
|
$dbexport = str_replace( $dbnamestring, "<database>\n\n <name>*dbname*", $dbexport );
|
|
|
|
$dbexport = str_replace( $dbtableprefixstring, "<table>\n\n <name>*dbprefix*", $dbexport );
|
|
|
|
// Write the new db export file
|
|
|
|
file_put_contents( $dbfile, $dbexport );
|
|
|
|
self::$zip->addFile( $dbfile, "dbexport.xml" );
|
|
|
|
// Add user data
|
|
|
|
foreach(OC_User::getUsers() as $user){
|
|
|
|
self::addDirToZip( $datadir . '/' . $user . '/', true, "/userdata/" );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'userfiles':
|
|
|
|
// Creates a zip with all of the users files
|
|
|
|
foreach(OC_User::getUsers() as $user){
|
|
|
|
self::addDirToZip( $datadir . '/' . $user . '/', true, "/" );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'system':
|
|
|
|
// Creates a zip with the owncloud system files
|
|
|
|
self::addDirToZip( OC::$SERVERROOT . '/', false, '/');
|
|
|
|
foreach (array(".git", "3rdparty", "apps", "core", "files", "l10n", "lib", "ocs", "search", "settings", "tests") as $dir) {
|
|
|
|
self::addDirToZip( OC::$SERVERROOT . '/' . $dir, true, "/");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// Add export info
|
|
|
|
self::addExportInfo();
|
2012-03-15 20:52:43 +00:00
|
|
|
// Close the zip
|
|
|
|
if( !self::closeZip() ){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return self::$zippath;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-03-16 21:09:36 +00:00
|
|
|
/**
|
|
|
|
* @breif adds a json file with infomation on the export to the zips root (used on import)
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
static private function addExportInfo(){
|
|
|
|
$info = array(
|
|
|
|
'ocversion' => OC_Util::getVersion(),
|
|
|
|
'exporttime' => time(),
|
|
|
|
'exportedby' => OC_User::getUser(),
|
|
|
|
'exporttype' => self::$exporttype
|
|
|
|
);
|
|
|
|
// Create json
|
|
|
|
$json = json_encode( $info );
|
|
|
|
$tmpfile = tempnam("/tmp", "oc_export_info_");
|
|
|
|
self::$tmpfiles[] = $tmpfile;
|
|
|
|
if( !file_put_contents( $tmpfile, $json ) ){
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
self::$zip->addFile( $tmpfile, "export_info.json" );
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-15 20:52:43 +00:00
|
|
|
/**
|
|
|
|
* @breif tried to finalise the zip
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
static private function closeZip(){
|
|
|
|
if( !self::$zip->close() ){
|
|
|
|
OC_Log::write('migration', 'Failed to save the zip with error: '.self::$zip->getStatusString(), OC_Log::ERROR);
|
2012-03-16 21:09:36 +00:00
|
|
|
self::cleanup();
|
2012-03-15 20:52:43 +00:00
|
|
|
return false;
|
|
|
|
} else {
|
2012-03-16 21:09:36 +00:00
|
|
|
OC_Log::write('migration', 'Export zip created ok', OC_Log::INFO);
|
|
|
|
self::cleanup();
|
2012-03-15 20:52:43 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-16 21:09:36 +00:00
|
|
|
/**
|
|
|
|
* @breif cleans up after the zip
|
|
|
|
*/
|
|
|
|
static private function cleanup(){
|
|
|
|
// Delete tmp files
|
|
|
|
foreach(self::$tmpfiles as $i){
|
|
|
|
unlink( $i );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-13 23:09:43 +00:00
|
|
|
/**
|
|
|
|
* @breif creates a zip user export
|
2012-03-14 16:43:06 +00:00
|
|
|
* @param optional $uid string user id of the user to export (defaults to current)
|
|
|
|
* @param optional $path string path to folder to create file in (with trailing slash) (defaults to current user's data dir)
|
|
|
|
* @return false on failure | string path on success
|
2012-03-13 23:09:43 +00:00
|
|
|
*/
|
2012-03-15 20:52:43 +00:00
|
|
|
static public function createUserExportFile( $uid=null, $path=null ){
|
2012-03-14 16:43:06 +00:00
|
|
|
// User passed?
|
|
|
|
$uid = is_null( $uid ) ? OC_User::getUser() : $uid ;
|
2012-03-13 23:09:43 +00:00
|
|
|
// Is a database user?
|
|
|
|
if( !OC_User_Database::userExists( $uid ) ){
|
|
|
|
OC_Log::write('migration', 'User: '.$uid.' is not in the database and so cannot be exported.', OC_Log::ERROR);
|
|
|
|
return false;
|
|
|
|
}
|
2012-03-14 16:43:06 +00:00
|
|
|
// Set the uid
|
2012-03-13 23:09:43 +00:00
|
|
|
self::$uid = $uid;
|
2012-03-14 16:43:06 +00:00
|
|
|
// Create the zip object
|
2012-03-13 23:09:43 +00:00
|
|
|
self::$zip = new ZipArchive;
|
2012-03-14 16:43:06 +00:00
|
|
|
// Calculate users data dir
|
|
|
|
$user = OC_User::getUser();
|
|
|
|
$userdatadir = OC_Config::getValue( 'datadirectory' ) . '/' . $user . '/';
|
|
|
|
// Calculate zip name
|
|
|
|
$zipname = "owncloud_userexport_" . $user . '_' . date("y-m-d_H-i-s") . ".zip";
|
|
|
|
// Calculate destination
|
|
|
|
if( !is_null( $path ) ){
|
|
|
|
// Path given
|
|
|
|
// Is a directory?
|
|
|
|
if( !is_dir( $path ) ){
|
2012-03-15 20:52:43 +00:00
|
|
|
OC_Log::write('migration', 'Path supplied to createUserExportFile() is not a directory', OC_Log::ERROR);
|
2012-03-14 16:43:06 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// Is writeable
|
|
|
|
if( !is_writeable( $path ) ){
|
2012-03-15 20:52:43 +00:00
|
|
|
OC_Log::write('migration', 'Path supplied to createUserExportFile() is not writeable', OC_Log::ERROR);
|
2012-03-14 16:43:06 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
self::$zippath = $path . $zipname;
|
|
|
|
} else {
|
|
|
|
// Save in users data dir
|
|
|
|
self::$zippath = $userdatadir . $zipname;
|
2012-03-13 23:09:43 +00:00
|
|
|
}
|
2012-03-15 20:52:43 +00:00
|
|
|
// Try to create the zip
|
|
|
|
if( !self::createZip() ){
|
|
|
|
return false;
|
2012-03-14 16:43:06 +00:00
|
|
|
}
|
|
|
|
// Export the app info
|
2012-03-13 23:09:43 +00:00
|
|
|
$info = json_encode( self::exportAppData() );
|
|
|
|
file_put_contents( $userdatadir . '/exportinfo.json', $info );
|
2012-03-14 16:43:06 +00:00
|
|
|
// Add the data dir to the zip
|
|
|
|
self::addDirToZip( $userdatadir );
|
2012-03-15 20:52:43 +00:00
|
|
|
// Close the zip
|
|
|
|
if( !self::closeZip() ){
|
|
|
|
return false;
|
2012-03-13 23:09:43 +00:00
|
|
|
}
|
2012-03-15 20:52:43 +00:00
|
|
|
// All good
|
2012-03-14 16:43:06 +00:00
|
|
|
return self::$zippath;
|
2012-03-13 23:09:43 +00:00
|
|
|
}
|
2012-03-15 20:52:43 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @breif tries to create the zip
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
static private function createZip(){
|
|
|
|
// Check if properties are set
|
|
|
|
if( !self::$zip || !self::$zippath ){
|
|
|
|
OC_Log::write('migration', 'createZip() called but $zip and/or $zippath have not been set', OC_Log::ERROR);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if ( self::$zip->open( self::$zippath, ZIPARCHIVE::CREATE ) !== TRUE ) {
|
|
|
|
OC_Log::write('migration', 'Failed to create the zip with error: '.self::$zip->getStatusString(), OC_Log::ERROR);
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
2012-03-13 23:09:43 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @breif adds a directory to the zip object
|
2012-03-14 16:43:06 +00:00
|
|
|
* @param $dir string path of the directory to add
|
|
|
|
* @param $recursive bool
|
|
|
|
* @param $internaldir string path of folder to add dir to in zip
|
|
|
|
* @return bool
|
2012-03-13 23:09:43 +00:00
|
|
|
*/
|
2012-03-14 16:43:06 +00:00
|
|
|
static private function addDirToZip($dir, $recursive=true, $internaldir='') {
|
|
|
|
$dirname = basename($dir);
|
|
|
|
self::$zip->addEmptyDir($internaldir . $dirname);
|
|
|
|
$internaldir.=$dirname.='/';
|
|
|
|
|
|
|
|
if ($dirhandle = opendir($dir)) {
|
2012-03-13 23:09:43 +00:00
|
|
|
while (false !== ( $file = readdir($dirhandle))) {
|
2012-03-14 16:43:06 +00:00
|
|
|
|
2012-03-13 23:09:43 +00:00
|
|
|
if (( $file != '.' ) && ( $file != '..' )) {
|
2012-03-14 16:43:06 +00:00
|
|
|
|
2012-03-13 23:09:43 +00:00
|
|
|
if (is_dir($dir . '/' . $file) && $recursive) {
|
2012-03-14 16:43:06 +00:00
|
|
|
self::addDirToZip($dir . '/' . $file, $recursive, $internaldir);
|
2012-03-13 23:09:43 +00:00
|
|
|
} elseif (is_file($dir . '/' . $file)) {
|
2012-03-14 16:43:06 +00:00
|
|
|
self::$zip->addFile($dir . '/' . $file, $internaldir . $file);
|
2012-03-13 23:09:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
closedir($dirhandle);
|
2012-03-14 16:43:06 +00:00
|
|
|
} else {
|
|
|
|
OC_Log::write('admin_export',"Was not able to open directory: " . $dir,OC_Log::ERROR);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2012-03-13 23:09:43 +00:00
|
|
|
|
2012-03-13 21:28:53 +00:00
|
|
|
/**
|
|
|
|
* @breif returns an array of apps that support migration
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
static public function getApps(){
|
|
|
|
$allapps = OC_App::getAllApps();
|
|
|
|
foreach($allapps as $app){
|
|
|
|
$path = OC::$SERVERROOT . '/apps/' . $app . '/lib/migrate.php';
|
|
|
|
if( file_exists( $path ) ){
|
|
|
|
$supportsmigration[] = $app;
|
|
|
|
}
|
|
|
|
}
|
2012-03-13 21:29:31 +00:00
|
|
|
return $supportsmigration;
|
2012-03-13 21:28:53 +00:00
|
|
|
}
|
|
|
|
|
2012-03-09 23:33:11 +00:00
|
|
|
/**
|
|
|
|
* @breif imports a new user
|
2012-03-12 21:41:32 +00:00
|
|
|
* @param $db string path to migration.db
|
2012-03-16 22:50:35 +00:00
|
|
|
* @param $info array of migration ino
|
2012-03-09 23:33:11 +00:00
|
|
|
* @param $uid optional uid to use
|
|
|
|
* @return bool if the import succedded
|
|
|
|
*/
|
2012-03-16 22:50:35 +00:00
|
|
|
public static function importAppData( $db, $info, $uid=false ){
|
2012-03-12 21:41:32 +00:00
|
|
|
|
2012-03-11 22:09:16 +00:00
|
|
|
if(!self::$uid){
|
|
|
|
OC_Log::write('migration','Tried to import without passing a uid',OC_Log::FATAL);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-03-12 21:41:32 +00:00
|
|
|
// Check if the db exists
|
|
|
|
if( file_exists( $db ) ){
|
|
|
|
// Connect to the db
|
|
|
|
if(!self::connectDB( $db )){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
OC_Log::write('migration','Migration.db not found at: '.$db, OC_Log::FATAL );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-03-16 22:50:35 +00:00
|
|
|
if( !is_array( $info ) ){
|
2012-03-13 16:21:17 +00:00
|
|
|
OC_Log::write('migration','$migrateinfo is not an array', OC_Log::FATAL);
|
2012-03-12 21:41:32 +00:00
|
|
|
return false;
|
2012-03-09 23:33:11 +00:00
|
|
|
}
|
|
|
|
|
2012-03-12 21:41:32 +00:00
|
|
|
// Set the user id
|
2012-03-16 22:50:35 +00:00
|
|
|
self::$uid = $info->migrateinfo->uid;
|
2012-03-12 21:41:32 +00:00
|
|
|
|
2012-03-16 22:50:35 +00:00
|
|
|
$apps = $info->app;
|
2012-03-12 21:41:32 +00:00
|
|
|
|
|
|
|
foreach( self::$providers as $provider){
|
|
|
|
// Is the app in the export?
|
|
|
|
if( array_key_exists( $provider->id, $apps ) ){
|
|
|
|
// Did it succeed?
|
|
|
|
if( $app[$provider->id] ){
|
|
|
|
// Then do the import
|
2012-03-16 22:50:35 +00:00
|
|
|
$provider->import( $info );
|
2012-03-12 21:41:32 +00:00
|
|
|
}
|
|
|
|
}
|
2012-03-09 23:33:11 +00:00
|
|
|
}
|
|
|
|
|
2012-03-10 18:18:58 +00:00
|
|
|
return true;
|
2012-03-09 23:33:11 +00:00
|
|
|
|
|
|
|
}
|
2012-03-03 17:30:21 +00:00
|
|
|
|
2012-03-09 23:33:11 +00:00
|
|
|
// @breif connects to migration.db, or creates if not found
|
2012-03-12 21:41:32 +00:00
|
|
|
// @param $db optional path to migration.db, defaults to user data dir
|
2012-03-09 23:33:11 +00:00
|
|
|
// @return bool whether the operation was successful
|
2012-03-13 16:21:17 +00:00
|
|
|
private static function connectDB( $dbpath=null ){
|
2012-03-11 22:09:16 +00:00
|
|
|
OC_Log::write('migration','connecting to migration.db for user: '.self::$uid,OC_Log::INFO);
|
|
|
|
// Fail if no user is set
|
|
|
|
if(!self::$uid){
|
|
|
|
OC_Log::write('migration','connectDB() called without self::$uid being set',OC_Log::INFO);
|
|
|
|
return false;
|
|
|
|
}
|
2012-03-09 23:33:11 +00:00
|
|
|
// Already connected
|
|
|
|
if(!self::$MDB2){
|
|
|
|
require_once('MDB2.php');
|
|
|
|
|
2012-03-10 18:18:58 +00:00
|
|
|
$datadir = OC_Config::getValue( "datadirectory", OC::$SERVERROOT."/data" );
|
2012-03-09 23:33:11 +00:00
|
|
|
|
2012-03-13 16:21:17 +00:00
|
|
|
self::$dbpath = $datadir.'/'.self::$uid.'/migration.db';//!is_null( $dbpath ) ? $dbpath : $datadir.'/'.self::$uid.'/migration.db';
|
|
|
|
|
2012-03-09 23:33:11 +00:00
|
|
|
// Prepare options array
|
|
|
|
$options = array(
|
|
|
|
'portability' => MDB2_PORTABILITY_ALL & (!MDB2_PORTABILITY_FIX_CASE),
|
|
|
|
'log_line_break' => '<br>',
|
|
|
|
'idxname_format' => '%s',
|
|
|
|
'debug' => true,
|
|
|
|
'quote_identifier' => true
|
|
|
|
);
|
|
|
|
$dsn = array(
|
2012-03-10 18:18:58 +00:00
|
|
|
'phptype' => 'sqlite3',
|
2012-03-13 16:21:17 +00:00
|
|
|
'database' => self::$dbpath,
|
2012-03-09 23:33:11 +00:00
|
|
|
'mode' => '0644'
|
|
|
|
);
|
2012-03-13 16:21:17 +00:00
|
|
|
|
2012-03-09 23:33:11 +00:00
|
|
|
// Try to establish connection
|
|
|
|
self::$MDB2 = MDB2::factory( $dsn, $options );
|
|
|
|
// Die if we could not connect
|
|
|
|
if( PEAR::isError( self::$MDB2 )){
|
2012-03-10 18:18:58 +00:00
|
|
|
die(self::$MDB2->getMessage());
|
2012-03-11 22:09:16 +00:00
|
|
|
OC_Log::write('migration', 'Failed to create/connect to migration.db',OC_Log::FATAL);
|
2012-03-09 23:33:11 +00:00
|
|
|
OC_Log::write('migration',self::$MDB2->getUserInfo(),OC_Log::FATAL);
|
|
|
|
OC_Log::write('migration',self::$MDB2->getMessage(),OC_Log::FATAL);
|
2012-03-03 17:30:21 +00:00
|
|
|
return false;
|
2012-03-10 18:18:58 +00:00
|
|
|
} else {
|
2012-03-03 17:30:21 +00:00
|
|
|
}
|
2012-03-09 23:33:11 +00:00
|
|
|
// We always, really always want associative arrays
|
|
|
|
self::$MDB2->setFetchMode(MDB2_FETCHMODE_ASSOC);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
2012-03-03 17:30:21 +00:00
|
|
|
|
2012-03-09 23:33:11 +00:00
|
|
|
// @breif prepares the db
|
|
|
|
// @param $query the sql query to prepare
|
2012-03-10 15:52:38 +00:00
|
|
|
public static function prepare( $query ){
|
2012-03-09 23:33:11 +00:00
|
|
|
|
|
|
|
// Optimize the query
|
|
|
|
$query = self::processQuery( $query );
|
|
|
|
|
|
|
|
// Optimize the query
|
|
|
|
$query = self::$MDB2->prepare( $query );
|
|
|
|
|
|
|
|
// Die if we have an error (error means: bad query, not 0 results!)
|
|
|
|
if( PEAR::isError( $query )) {
|
|
|
|
$entry = 'DB Error: "'.$result->getMessage().'"<br />';
|
|
|
|
$entry .= 'Offending command was: '.$query.'<br />';
|
|
|
|
OC_Log::write('migration',$entry,OC_Log::FATAL);
|
|
|
|
return false;
|
|
|
|
} else {
|
2012-03-10 18:18:58 +00:00
|
|
|
return $query;
|
2012-03-09 23:33:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2012-03-03 17:30:21 +00:00
|
|
|
|
2012-03-09 23:33:11 +00:00
|
|
|
// @breif processes the db query
|
|
|
|
// @param $query the query to process
|
|
|
|
// @return string of processed query
|
|
|
|
private static function processQuery( $query ){
|
2012-03-03 17:30:21 +00:00
|
|
|
|
2012-03-09 23:33:11 +00:00
|
|
|
self::connectDB();
|
|
|
|
$prefix = '';
|
|
|
|
|
|
|
|
$query = str_replace( '`', '\'', $query );
|
|
|
|
$query = str_replace( 'NOW()', 'datetime(\'now\')', $query );
|
|
|
|
$query = str_replace( 'now()', 'datetime(\'now\')', $query );
|
|
|
|
|
|
|
|
// replace table name prefix
|
|
|
|
$query = str_replace( '*PREFIX*', $prefix, $query );
|
|
|
|
|
|
|
|
return $query;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-03-10 15:52:38 +00:00
|
|
|
// @brief copys rows to migration.db from the main database
|
|
|
|
// @param $options array of options.
|
|
|
|
// @return bool
|
|
|
|
public static function copyRows( $options ){
|
|
|
|
if( !array_key_exists( 'table', $options ) ){
|
|
|
|
return false;
|
|
|
|
}
|
2012-03-11 22:09:16 +00:00
|
|
|
|
|
|
|
$return = array();
|
|
|
|
|
2012-03-10 15:52:38 +00:00
|
|
|
// Need to include 'where' in the query?
|
|
|
|
if( array_key_exists( 'matchval', $options ) && array_key_exists( 'matchcol', $options ) ){
|
2012-03-10 18:18:58 +00:00
|
|
|
|
|
|
|
// If only one matchval, create an array
|
|
|
|
if(!is_array($options['matchval'])){
|
|
|
|
$options['matchval'] = array( $options['matchval'] );
|
|
|
|
}
|
|
|
|
|
2012-03-10 15:52:38 +00:00
|
|
|
foreach( $options['matchval'] as $matchval ){
|
|
|
|
// Run the query for this match value (where x = y value)
|
|
|
|
$query = OC_DB::prepare( "SELECT * FROM *PREFIX*" . $options['table'] . " WHERE " . $options['matchcol'] . " LIKE ?" );
|
|
|
|
$results = $query->execute( array( $matchval ) );
|
2012-03-11 22:13:50 +00:00
|
|
|
$newreturns = self::insertData( $results, $options );
|
|
|
|
$return = array_merge( $return, $newreturns );
|
2012-03-10 15:52:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
// Just get everything
|
|
|
|
$query = OC_DB::prepare( "SELECT * FROM *PREFIX*" . $options['table'] );
|
|
|
|
$results = $query->execute();
|
2012-03-11 22:09:16 +00:00
|
|
|
$return = self::insertData( $results, $options );
|
2012-03-10 15:52:38 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-03-11 22:09:16 +00:00
|
|
|
return $return;
|
2012-03-10 15:52:38 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// @breif saves a sql data set into migration.db
|
|
|
|
// @param $data a sql data set returned from self::prepare()->query()
|
|
|
|
// @param $options array of copyRows options
|
|
|
|
// @return void
|
|
|
|
private static function insertData( $data, $options ){
|
2012-03-11 22:09:16 +00:00
|
|
|
$return = array();
|
2012-03-10 18:18:58 +00:00
|
|
|
while( $row = $data->fetchRow() ){
|
2012-03-10 15:52:38 +00:00
|
|
|
// Now save all this to the migration.db
|
2012-03-11 22:09:16 +00:00
|
|
|
$fields = array();
|
|
|
|
$values = array();
|
2012-03-10 15:52:38 +00:00
|
|
|
foreach($row as $field=>$value){
|
|
|
|
$fields[] = $field;
|
|
|
|
$values[] = $value;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate some sql
|
2012-03-10 18:18:58 +00:00
|
|
|
$sql = "INSERT INTO `" . $options['table'] . '` ( `';
|
2012-03-10 15:52:38 +00:00
|
|
|
$fieldssql = implode( '`, `', $fields );
|
|
|
|
$sql .= $fieldssql . "` ) VALUES( ";
|
2012-03-10 18:18:58 +00:00
|
|
|
$valuessql = substr( str_repeat( '?, ', count( $fields ) ),0,-2 );
|
2012-03-10 15:52:38 +00:00
|
|
|
$sql .= $valuessql . " )";
|
|
|
|
// Make the query
|
|
|
|
$query = self::prepare( $sql );
|
2012-03-10 18:18:58 +00:00
|
|
|
if(!$query){
|
|
|
|
OC_Log::write('migration','Invalid sql produced: '.$sql,OC_Log::FATAL);
|
2012-03-11 22:09:16 +00:00
|
|
|
return false;
|
|
|
|
exit();
|
2012-03-10 18:18:58 +00:00
|
|
|
} else {
|
|
|
|
$query->execute( $values );
|
2012-03-11 22:09:16 +00:00
|
|
|
// Do we need to return some values?
|
|
|
|
if( array_key_exists( 'idcol', $options ) ){
|
|
|
|
// Yes we do
|
|
|
|
$return[] = $row[$options['idcol']];
|
|
|
|
} else {
|
|
|
|
// Take a guess and return the first field :)
|
|
|
|
$return[] = reset($row);
|
|
|
|
}
|
2012-03-10 18:18:58 +00:00
|
|
|
}
|
2012-03-10 15:52:38 +00:00
|
|
|
}
|
2012-03-11 22:09:16 +00:00
|
|
|
return $return;
|
2012-03-10 15:52:38 +00:00
|
|
|
}
|
|
|
|
|
2012-03-09 23:33:11 +00:00
|
|
|
// @breif creates the tables in migration.db from an apps database.xml
|
|
|
|
// @param $appid string id of the app
|
|
|
|
// @return bool whether the operation was successful
|
|
|
|
private static function createAppTables( $appid ){
|
|
|
|
|
2012-03-12 18:40:14 +00:00
|
|
|
if(!self::connectScheme()){
|
2012-03-03 17:30:21 +00:00
|
|
|
return false;
|
2012-02-03 20:32:06 +00:00
|
|
|
}
|
2012-03-12 18:40:14 +00:00
|
|
|
|
|
|
|
// There is a database.xml file
|
|
|
|
$content = file_get_contents( OC::$SERVERROOT . '/apps/' . $appid . '/appinfo/database.xml' );
|
|
|
|
|
|
|
|
$file2 = 'static://db_scheme';
|
|
|
|
$content = str_replace( '*dbname*', self::$uid.'/migration', $content );
|
|
|
|
$content = str_replace( '*dbprefix*', '', $content );
|
|
|
|
|
|
|
|
$xml = new SimpleXMLElement($content);
|
|
|
|
foreach($xml->table as $table){
|
|
|
|
$tables[] = (string)$table->name;
|
|
|
|
}
|
|
|
|
|
|
|
|
file_put_contents( $file2, $content );
|
|
|
|
|
|
|
|
// Try to create tables
|
|
|
|
$definition = self::$schema->parseDatabaseDefinitionFile( $file2 );
|
|
|
|
|
|
|
|
unlink( $file2 );
|
|
|
|
|
|
|
|
// Die in case something went wrong
|
|
|
|
if( $definition instanceof MDB2_Schema_Error ){
|
|
|
|
OC_Log::write('migration','Failed to parse database.xml for: '.$appid,OC_Log::FATAL);
|
|
|
|
OC_Log::write('migration',$definition->getMessage().': '.$definition->getUserInfo(),OC_Log::FATAL);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$definition['overwrite'] = true;
|
|
|
|
|
|
|
|
$ret = self::$schema->createDatabase( $definition );
|
|
|
|
// Die in case something went wrong
|
|
|
|
|
|
|
|
if( $ret instanceof MDB2_Error ){
|
|
|
|
OC_Log::write('migration','Failed to create tables for: '.$appid,OC_Log::FATAL);
|
|
|
|
OC_Log::write('migration',$ret->getMessage().': '.$ret->getUserInfo(),OC_Log::FATAL);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return $tables;
|
|
|
|
|
2012-02-03 20:32:06 +00:00
|
|
|
}
|
|
|
|
|
2012-03-09 23:33:11 +00:00
|
|
|
|
2012-03-14 16:43:06 +00:00
|
|
|
/**
|
2012-03-09 23:33:11 +00:00
|
|
|
* @brief connects to a MDB2 database scheme
|
|
|
|
* @returns true/false
|
|
|
|
*
|
|
|
|
* Connects to a MDB2 database scheme
|
2012-03-03 17:30:21 +00:00
|
|
|
*/
|
2012-03-09 23:33:11 +00:00
|
|
|
private static function connectScheme(){
|
|
|
|
// We need a mdb2 database connection
|
|
|
|
self::connectDB();
|
|
|
|
self::$MDB2->loadModule( 'Manager' );
|
|
|
|
self::$MDB2->loadModule( 'Reverse' );
|
|
|
|
|
|
|
|
// Connect if this did not happen before
|
|
|
|
if( !self::$schema ){
|
|
|
|
require_once('MDB2/Schema.php');
|
|
|
|
self::$schema=MDB2_Schema::factory( self::$MDB2 );
|
2012-03-03 17:30:21 +00:00
|
|
|
}
|
|
|
|
|
2012-03-09 23:33:11 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// @breif creates a new user in the database
|
|
|
|
// @param $uid string user_id of the user to be created
|
|
|
|
// @param $hash string hash of the user to be created
|
|
|
|
// @return bool result of user creation
|
|
|
|
private static function createUser( $uid, $hash ){
|
|
|
|
|
|
|
|
// Check if userid exists
|
|
|
|
if(OC_User::userExists( $uid )){
|
|
|
|
return false;
|
2012-03-03 17:30:21 +00:00
|
|
|
}
|
|
|
|
|
2012-03-09 23:33:11 +00:00
|
|
|
// Create the user
|
|
|
|
$query = OC_DB::prepare( "INSERT INTO `*PREFIX*users` ( `uid`, `password` ) VALUES( ?, ? )" );
|
|
|
|
$result = $query->execute( array( $uid, $data['hash']));
|
2012-03-12 21:41:32 +00:00
|
|
|
if( !$result ){
|
|
|
|
OC_Log::write('migration', 'Failed to create the new user "'.$uid."");
|
|
|
|
}
|
2012-03-09 23:33:11 +00:00
|
|
|
return $result ? true : false;
|
|
|
|
|
|
|
|
}
|
2012-03-16 21:09:36 +00:00
|
|
|
|
2012-02-03 20:32:06 +00:00
|
|
|
}
|