mirror of
				https://github.com/Alamantus/Lexiconga.git
				synced 2025-10-26 14:06:48 +01:00 
			
		
		
		
	Write and test migration script; Update readme
This commit is contained in:
		
							parent
							
								
									102877aefa
								
							
						
					
					
						commit
						118b5c2d1a
					
				
					 3 changed files with 284 additions and 1 deletions
				
			
		
							
								
								
									
										1
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							|  | @ -4,3 +4,4 @@ dist/ | |||
| vendor/ | ||||
| 
 | ||||
| src/php/api/config.php | ||||
| src/php/api/migrate.php | ||||
|  | @ -25,3 +25,11 @@ It's less useful, but `npm run serve-frontend-only` will bundle and serve _only_ | |||
| ## Production | ||||
| 
 | ||||
| `npm run bundle` bundles and minifies the frontend stuff and also copies the backend stuff to `dist`. Be sure to run `npm run clear` to delete the contents of `dist` and `.cache` before using `npm run bundle` to make sure you don't get old dev versions of the bundled code included in your upload. | ||||
| 
 | ||||
| ## Migration | ||||
| 
 | ||||
| There is a script called `src/php/api/migrate.php.changeme` that can be used to help with the migration process from a `version1` Lexiconga database into a `master` database. **Note:** Migration is intended only for migrating from an old server to a freshly-installed/empty new database. To use this, copy `src/php/api/migrate.php.changeme` to `migrate.php` somewhere in the `version1` project (probably in `/php`) and copy the same to `/api/migrate.php` in your `master` project, making sure that all the variables for referencing the databases are correct. | ||||
| 
 | ||||
| Visit `migrate.php` on your `version1` server with `?outgoing=true` set in order to begin the transfer. The other server's `migrate.php` will receive an "incoming" request multiple times, and your screen will display messages as it works. | ||||
| 
 | ||||
| _DELETE THESE `migrate.php` FILES IMMEDIATELY AFTER MIGRATION IS COMPLETE!_. | ||||
|  |  | |||
							
								
								
									
										274
									
								
								src/php/api/migrate.php.changeme
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										274
									
								
								src/php/api/migrate.php.changeme
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,274 @@ | |||
| <?php | ||||
| $incoming = isset($_POST['incoming']) ? json_decode($_POST['incoming'], true) : false; | ||||
| $outgoing = isset($_GET['outgoing']); | ||||
| 
 | ||||
| header('Access-Control-Allow-Origin: *'); | ||||
| header('Expires: Sun, 01 Nov 2015 22:46:51 GMT'); | ||||
| header("Cache-Control: no-store, no-cache, must-revalidate, max-age=0"); | ||||
| header("Cache-Control: post-check=0, pre-check=0", false); | ||||
| header("Pragma: no-cache"); | ||||
| 
 | ||||
| define("VERSION1_DATABASE_SERVERNAME", "host"); | ||||
| define("VERSION1_DATABASE_USERNAME", "username"); | ||||
| define("VERSION1_DATABASE_PASSWORD", "password"); | ||||
| define("VERSION1_DATABASE_NAME", "database"); | ||||
| define("MIGRATE_TO", "remote_url"); | ||||
| define("NEW_DATABASE_SERVERNAME", "host"); | ||||
| define("NEW_DATABASE_USERNAME", "username"); | ||||
| define("NEW_DATABASE_PASSWORD", "password"); | ||||
| define("NEW_DATABASE_NAME", "database"); | ||||
| 
 | ||||
| if ($outgoing) { | ||||
|   echo '<pre>' . PHP_EOL; | ||||
|   echo 'Starting migration from ' . VERSION1_DATABASE_SERVERNAME . ' db ' . VERSION1_DATABASE_NAME . PHP_EOL . 'To ' . MIGRATE_TO . PHP_EOL . PHP_EOL; | ||||
| 
 | ||||
|   $dbconnection = new PDO('mysql:host=' . VERSION1_DATABASE_SERVERNAME . ';dbname=' . VERSION1_DATABASE_NAME . ';charset=utf8', VERSION1_DATABASE_USERNAME, VERSION1_DATABASE_PASSWORD); | ||||
| 	$dbconnection->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION); | ||||
| 	// $dbconnection->setAttribute(PDO::ATTR_EMULATE_PREPARES, true); | ||||
| 	$dbconnection->setAttribute(PDO::ATTR_DEFAULT_FETCH_MODE, PDO::FETCH_ASSOC); | ||||
|   try { | ||||
|     $users_stmt = $dbconnection->prepare('SELECT * FROM `users`'); | ||||
|     $users_stmt->execute(); | ||||
|     if ($users_stmt) { | ||||
|       $users = $users_stmt->fetchAll(); | ||||
|       if ($users !== false) { | ||||
|         echo count($users) . ' Users' . PHP_EOL; | ||||
| 
 | ||||
|         foreach($users as $user) { | ||||
|           $send_user_response = send('user', $user); | ||||
| 
 | ||||
|           if ($send_user_response === "GOT USER") { | ||||
|             echo PHP_EOL . 'User ' . $user['id'] . ' sent successfully' . PHP_EOL; | ||||
| 
 | ||||
|             $dictionaries_stmt = $dbconnection->prepare('SELECT * FROM `dictionaries` WHERE `user`=?'); | ||||
|             $dictionaries_stmt->execute(array(intval($user['id']))); | ||||
|             if ($dictionaries_stmt) { | ||||
|               $dictionaries = $dictionaries_stmt->fetchAll(); | ||||
|               if ($dictionaries !== false) { | ||||
|                 echo 'User ' . $user['id'] . ' has ' . count($dictionaries) . ' Dictionaries' . PHP_EOL; | ||||
| 
 | ||||
|                 $send_dictionaries_response = send('dictionaries', $dictionaries, array('user' => $user['id'])); | ||||
| 
 | ||||
|                 if ($send_dictionaries_response === 'GOT DICTIONARIES') { | ||||
|                   echo 'Dictionaries sent successfully' . PHP_EOL; | ||||
| 
 | ||||
|                   foreach($dictionaries as $dictionary) { | ||||
|                     $words_stmt = $dbconnection->prepare('SELECT * FROM `words` WHERE `dictionary`=?'); | ||||
|                     $words_stmt->execute(array(intval($dictionary['id']))); | ||||
|                     if ($words_stmt) { | ||||
|                       $words = $words_stmt->fetchAll(); | ||||
|                       if ($words !== false) { | ||||
|                         $send_words_response = send('words', $words); | ||||
| 
 | ||||
|                         if ($send_words_response === 'GOT WORDS') { | ||||
|                           echo 'Words for Dictionary ' . $dictionary['id'] . ' sent successfully' . PHP_EOL; | ||||
|                         } else { | ||||
|                           echo var_export($send_words_response, true) . PHP_EOL . PHP_EOL; | ||||
|                         } | ||||
|                       } else { | ||||
|                         echo '$words_stmt->fetchAll() failed.' . PHP_EOL . var_export($words_stmt->errorInfo(), true) . PHP_EOL . PHP_EOL; | ||||
|                       } | ||||
|                     } else { | ||||
|                       echo '$words_stmt failed' . PHP_EOL . PHP_EOL; | ||||
|                     } | ||||
|                   } | ||||
|                 } else { | ||||
|                   echo var_export($send_dictionaries_response, true) . PHP_EOL . PHP_EOL; | ||||
|                 } | ||||
|               } else { | ||||
|                 echo '$dictionaries_stmt->fetchAll() failed.' . PHP_EOL . var_export($dictionaries_stmt->errorInfo(), true) . PHP_EOL . PHP_EOL; | ||||
|               } | ||||
|             } else { | ||||
|               echo '$dictionaries_stmt failed' . PHP_EOL . PHP_EOL; | ||||
|             } | ||||
|           } else { | ||||
|             var_dump($send_user_response) . PHP_EOL . PHP_EOL; | ||||
|           } | ||||
|         } | ||||
|       } else { | ||||
|         echo '$users_stmt->fetchAll() failed.' . PHP_EOL . var_export($users_stmt->errorInfo(), true) . PHP_EOL . PHP_EOL; | ||||
|       } | ||||
|     } else { | ||||
|       echo '$users_stmt failed' . PHP_EOL . PHP_EOL; | ||||
|     } | ||||
|   } | ||||
|   catch (PDOException $ex) { | ||||
|     echo var_export($ex, true) . PHP_EOL . PHP_EOL; | ||||
|   } | ||||
| 
 | ||||
|   echo '</pre>'; | ||||
| } else if ($incoming !== false) { | ||||
|   $type = isset($_POST['type']) ? $_POST['type'] : false; | ||||
| 
 | ||||
|   if ($type !== false) { | ||||
|     $dbconnection = new PDO('mysql:host=' . NEW_DATABASE_SERVERNAME . ';dbname=' . NEW_DATABASE_NAME . ';charset=utf8', NEW_DATABASE_USERNAME, NEW_DATABASE_PASSWORD); | ||||
|     $dbconnection->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION); | ||||
|     switch($type) { | ||||
|       case 'user': { | ||||
|         if ($incoming === null) { | ||||
|           echo 'Invalid JSON'; | ||||
|         } else { | ||||
|           $user_stmt = $dbconnection->prepare('INSERT INTO `users`(`id`, `email`, `old_password`, `password`, `public_name`, `current_dictionary`, `allow_email`, `last_login`, `password_reset_code`, `password_reset_date`, `created_on`) VALUES (?,?,?,?,?,?,?,?,?,?,?)'); | ||||
|           $user_stmt->execute(array( | ||||
|             intval($incoming['id']), | ||||
|             $incoming['email'], | ||||
|             $incoming['password'], | ||||
|             null, | ||||
|             $incoming['public_name'], | ||||
|             intval($incoming['current_dictionary']), | ||||
|             intval($incoming['allow_email']), | ||||
|             $incoming['last_login'], | ||||
|             null, | ||||
|             null, | ||||
|             $incoming['created_on'], | ||||
|           )); | ||||
|           if ($user_stmt) { | ||||
|             echo 'GOT USER'; | ||||
|           } else { | ||||
|             var_dump($user_stmt->errorInfo()); | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|       case 'dictionaries': { | ||||
|         if ($incoming === null) { | ||||
|           echo 'Invalid JSON'; | ||||
|         } else { | ||||
|           if (count($incoming) === 0) { | ||||
|             echo 'Need new dictionary for user ' . $_POST['user'] . ': '; | ||||
|              | ||||
|             $new_id = mt_rand(1000, 999999999); | ||||
| 
 | ||||
|             $user = intval($_POST['user']); | ||||
|             $insert_dictionary_query = "INSERT INTO dictionaries (id, user, description, created_on) VALUES (?, ?, ?, ?)"; | ||||
|             $insert_dictionary_stmt = $dbconnection->prepare($insert_dictionary_query); | ||||
|             $insert_dictionary = $insert_dictionary_stmt->execute(array($new_id, $user, 'A new dictionary.', time())); | ||||
| 
 | ||||
|             if ($insert_dictionary) { | ||||
|               $insert_linguistics_query = "INSERT INTO dictionary_linguistics (dictionary, parts_of_speech, exceptions, orthography_notes, grammar_notes) VALUES ($new_id, ?, ?, ?, ?)"; | ||||
|               $insert_linguistics_stmt = $dbconnection->prepare($insert_linguistics_query); | ||||
|               $insert_linguistics = $insert_linguistics_stmt->execute(array('Noun,Adjective,Verb', '', '', '')); | ||||
| 
 | ||||
|               if ($insert_linguistics === true) { | ||||
|                 $update_query = 'UPDATE users SET current_dictionary=? WHERE id=?'; | ||||
|                 $update_stmt = $dbconnection->prepare($update_query); | ||||
|                 $update = $update_stmt->execute(array($new_id, $user)); | ||||
|                 if ($update) { | ||||
|                   echo 'CREATED DICTIONARY'; | ||||
|                 } else { | ||||
|                   var_dump($update_stmt); | ||||
|                 } | ||||
|               } else { | ||||
|                 var_dump($insert_linguistics_stmt->errorInfo()); | ||||
|               } | ||||
|             } else { | ||||
|               var_dump($insert_dictionary_stmt->errorInfo()); | ||||
|             } | ||||
|           } else { | ||||
|             $dictionaries_query = 'INSERT INTO `dictionaries`(`id`, `user`, `name`, `specification`, `description`, `allow_duplicates`, `case_sensitive`, `sort_by_definition`, `theme`, `is_public`, `last_updated`, `created_on`) VALUES '; | ||||
|             $linguistics_query = "INSERT INTO dictionary_linguistics (dictionary, parts_of_speech, exceptions, orthography_notes, grammar_notes) VALUES "; | ||||
|             $dictionaries_params = array(); | ||||
|             $linguistics_params = array(); | ||||
|             foreach($incoming as $dictionary) { | ||||
|               $dictionaries_query .= '(?,?,?,?,?,?,?,?,?,?,?,?), '; | ||||
|               // `id`, `user`, `name`, `description`, `words`, `next_word_id`, `allow_duplicates`, `case_sensitive`, `parts_of_speech`, `sort_by_equivalent`, `is_complete`, `is_public`, `last_updated`, `created_on` | ||||
|               $dictionaries_params[] = intval($dictionary['id']); | ||||
|               $dictionaries_params[] = intval($dictionary['user']); | ||||
|               $dictionaries_params[] = $dictionary['name']; | ||||
|               $dictionaries_params[] = 'Dictionary'; | ||||
|               $dictionaries_params[] = fixStupidOldNonsense($dictionary['description']); | ||||
|               $dictionaries_params[] = intval($dictionary['allow_duplicates']); | ||||
|               $dictionaries_params[] = intval($dictionary['case_sensitive']); | ||||
|               $dictionaries_params[] = intval($dictionary['sort_by_equivalent']); | ||||
|               $dictionaries_params[] = 'default'; | ||||
|               $dictionaries_params[] = intval($dictionary['is_public']); | ||||
|               $dictionaries_params[] = $dictionary['last_updated'] ? strtotime($dictionary['last_updated']) : null; | ||||
|               $dictionaries_params[] = strtotime($dictionary['created_on']); | ||||
| 
 | ||||
|               $linguistics_query .= '(?, ?, ?, ?, ?), '; | ||||
|               $linguistics_params[] = intval($dictionary['id']); | ||||
|               $linguistics_params[] = $dictionary['parts_of_speech']; | ||||
|               $linguistics_params[] = ''; | ||||
|               $linguistics_params[] = ''; | ||||
|               $linguistics_params[] = ''; | ||||
|             } | ||||
|             $dictionaries_stmt = $dbconnection->prepare(trim($dictionaries_query, ', ')); | ||||
|             $dictionaries_stmt->execute($dictionaries_params); | ||||
|             if ($dictionaries_stmt) { | ||||
|               $linguistics_stmt = $dbconnection->prepare(trim($linguistics_query, ', ')); | ||||
|               $linguistics_stmt->execute($linguistics_params); | ||||
|               if ($linguistics_stmt) { | ||||
|                 echo 'GOT DICTIONARIES'; | ||||
|               } else { | ||||
|                 var_dump($linguistics_stmt->errorInfo()); | ||||
|               } | ||||
|             } else { | ||||
|               var_dump($dictionaries_stmt->errorInfo()); | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|       case 'words': { | ||||
|         if ($incoming === null) { | ||||
|           echo 'Invalid JSON'; | ||||
|         } else { | ||||
|           if (count($incoming) > 0) { | ||||
|             $words_query = 'INSERT INTO `words`(`dictionary`, `word_id`, `name`, `pronunciation`, `part_of_speech`, `definition`, `details`, `last_updated`, `created_on`) VALUES '; | ||||
|             $words_params = array(); | ||||
|             foreach($incoming as $word) { | ||||
|               $words_query .= '(?,?,?,?,?,?,?,?,?), '; | ||||
|               // `dictionary`, `word_id`, `name`, `pronunciation`, `part_of_speech`, `simple_definition`, `long_definition`, `last_updated`, `created_on` | ||||
|               $words_params[] = intval($word['dictionary']); | ||||
|               $words_params[] = intval($word['word_id']); | ||||
|               $words_params[] = $word['name']; | ||||
|               $words_params[] = $word['pronunciation']; | ||||
|               $words_params[] = $word['part_of_speech']; | ||||
|               $words_params[] = $word['simple_definition']; | ||||
|               $words_params[] = fixStupidOldNonsense($word['long_definition']); | ||||
|               $words_params[] = $word['last_updated'] ? strtotime($word['last_updated']) : null; | ||||
|               $words_params[] = strtotime($word['created_on']); | ||||
|             } | ||||
|             $words_stmt = $dbconnection->prepare(trim($words_query, ', ')); | ||||
|             $words_stmt->execute($words_params); | ||||
|             if ($words_stmt) { | ||||
|               echo 'GOT WORDS'; | ||||
|             } else { | ||||
|               var_dump($words_stmt->errorInfo()); | ||||
|             } | ||||
|           } else { | ||||
|             echo 'NO WORDS'; | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } else { | ||||
|   echo 'no' . PHP_EOL . PHP_EOL; | ||||
| } | ||||
| 
 | ||||
| function send($type, $data, $additional_data = array()) { | ||||
|   $send_data = array_merge(array('type' => $type, 'incoming' => json_encode($data)), $additional_data); | ||||
| 
 | ||||
|   $ch = curl_init(); | ||||
|   curl_setopt($ch, CURLOPT_URL, MIGRATE_TO . '/api/migrate.php'); | ||||
|   curl_setopt($ch, CURLOPT_POST, 1); | ||||
|   curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query($send_data)); | ||||
| 
 | ||||
|   curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); | ||||
| 
 | ||||
|   $server_output = curl_exec($ch); | ||||
| 
 | ||||
|   curl_close ($ch); | ||||
| 
 | ||||
|   return $server_output; | ||||
| } | ||||
| 
 | ||||
| function fixStupidOldNonsense($markdown) { | ||||
|   $markdown = str_replace('"', '"', $markdown); | ||||
|   $markdown = str_replace(''', "'", $markdown); | ||||
|   $markdown = str_replace('\', '\\', $markdown); | ||||
|   $markdown = str_replace('<br>', '\\n', $markdown); | ||||
|   return $markdown; | ||||
| } | ||||
		Loading…
	
	Add table
		
		Reference in a new issue