tbl_replace.php 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403
  1. <?php
  2. /* vim: set expandtab sw=4 ts=4 sts=4: */
  3. /**
  4. * Manipulation of table data like inserting, replacing and updating
  5. *
  6. * Usally called as form action from tbl_change.php to insert or update table rows
  7. *
  8. * @todo 'edit_next' tends to not work as expected if used ...
  9. * at least there is no order by it needs the original query
  10. * and the row number and than replace the LIMIT clause
  11. *
  12. * @package PhpMyAdmin
  13. */
  14. /**
  15. * Gets some core libraries
  16. */
  17. require_once 'libraries/common.inc.php';
  18. /**
  19. * functions implementation for this script
  20. */
  21. require_once 'libraries/insert_edit.lib.php';
  22. // Check parameters
  23. PMA_Util::checkParameters(array('db', 'table', 'goto'));
  24. $GLOBALS['dbi']->selectDb($GLOBALS['db']);
  25. /**
  26. * Initializes some variables
  27. */
  28. $goto_include = false;
  29. $response = PMA_Response::getInstance();
  30. $header = $response->getHeader();
  31. $scripts = $header->getScripts();
  32. $scripts->addFile('makegrid.js');
  33. // Needed for generation of Inline Edit anchors
  34. $scripts->addFile('sql.js');
  35. $scripts->addFile('indexes.js');
  36. $scripts->addFile('gis_data_editor.js');
  37. // check whether insert row mode, if so include tbl_change.php
  38. PMA_isInsertRow();
  39. $after_insert_actions = array('new_insert', 'same_insert', 'edit_next');
  40. if (isset($_REQUEST['after_insert'])
  41. && in_array($_REQUEST['after_insert'], $after_insert_actions)
  42. ) {
  43. $url_params['after_insert'] = $_REQUEST['after_insert'];
  44. if (isset($_REQUEST['where_clause'])) {
  45. foreach ($_REQUEST['where_clause'] as $one_where_clause) {
  46. if ($_REQUEST['after_insert'] == 'same_insert') {
  47. $url_params['where_clause'][] = $one_where_clause;
  48. } elseif ($_REQUEST['after_insert'] == 'edit_next') {
  49. PMA_setSessionForEditNext($one_where_clause);
  50. }
  51. }
  52. }
  53. }
  54. //get $goto_include for different cases
  55. $goto_include = PMA_getGotoInclude($goto_include);
  56. // Defines the url to return in case of failure of the query
  57. $err_url = PMA_getErrorUrl($url_params);
  58. /**
  59. * Prepares the update/insert of a row
  60. */
  61. list($loop_array, $using_key, $is_insert, $is_insertignore)
  62. = PMA_getParamsForUpdateOrInsert();
  63. $query = array();
  64. $value_sets = array();
  65. $func_no_param = array(
  66. 'CONNECTION_ID',
  67. 'CURRENT_USER',
  68. 'CURDATE',
  69. 'CURTIME',
  70. 'CURRENT_DATE',
  71. 'CURRENT_TIME',
  72. 'DATABASE',
  73. 'LAST_INSERT_ID',
  74. 'NOW',
  75. 'PI',
  76. 'RAND',
  77. 'SYSDATE',
  78. 'UNIX_TIMESTAMP',
  79. 'USER',
  80. 'UTC_DATE',
  81. 'UTC_TIME',
  82. 'UTC_TIMESTAMP',
  83. 'UUID',
  84. 'UUID_SHORT',
  85. 'VERSION',
  86. );
  87. $func_optional_param = array(
  88. 'RAND',
  89. 'UNIX_TIMESTAMP',
  90. );
  91. $gis_from_text_functions = array(
  92. 'GeomFromText',
  93. 'GeomCollFromText',
  94. 'LineFromText',
  95. 'MLineFromText',
  96. 'PointFromText',
  97. 'MPointFromText',
  98. 'PolyFromText',
  99. 'MPolyFromText',
  100. );
  101. $gis_from_wkb_functions = array(
  102. 'GeomFromWKB',
  103. 'GeomCollFromWKB',
  104. 'LineFromWKB',
  105. 'MLineFromWKB',
  106. 'PointFromWKB',
  107. 'MPointFromWKB',
  108. 'PolyFromWKB',
  109. 'MPolyFromWKB',
  110. );
  111. // to create an object of PMA_File class
  112. require_once './libraries/File.class.php';
  113. $query_fields = array();
  114. foreach ($loop_array as $rownumber => $where_clause) {
  115. // skip fields to be ignored
  116. if (! $using_key && isset($_REQUEST['insert_ignore_' . $where_clause])) {
  117. continue;
  118. }
  119. // Defines the SET part of the sql query
  120. $query_values = array();
  121. // Map multi-edit keys to single-level arrays, dependent on how we got the fields
  122. $multi_edit_colummns
  123. = isset($_REQUEST['fields']['multi_edit'][$rownumber])
  124. ? $_REQUEST['fields']['multi_edit'][$rownumber]
  125. : array();
  126. $multi_edit_columns_name
  127. = isset($_REQUEST['fields_name']['multi_edit'][$rownumber])
  128. ? $_REQUEST['fields_name']['multi_edit'][$rownumber]
  129. : null;
  130. $multi_edit_columns_prev
  131. = isset($_REQUEST['fields_prev']['multi_edit'][$rownumber])
  132. ? $_REQUEST['fields_prev']['multi_edit'][$rownumber]
  133. : null;
  134. $multi_edit_funcs
  135. = isset($_REQUEST['funcs']['multi_edit'][$rownumber])
  136. ? $_REQUEST['funcs']['multi_edit'][$rownumber]
  137. : null;
  138. $multi_edit_salt
  139. = isset($_REQUEST['salt']['multi_edit'][$rownumber])
  140. ? $_REQUEST['salt']['multi_edit'][$rownumber]
  141. :null;
  142. $multi_edit_columns_type
  143. = isset($_REQUEST['fields_type']['multi_edit'][$rownumber])
  144. ? $_REQUEST['fields_type']['multi_edit'][$rownumber]
  145. : null;
  146. $multi_edit_columns_null
  147. = isset($_REQUEST['fields_null']['multi_edit'][$rownumber])
  148. ? $_REQUEST['fields_null']['multi_edit'][$rownumber]
  149. : null;
  150. $multi_edit_columns_null_prev
  151. = isset($_REQUEST['fields_null_prev']['multi_edit'][$rownumber])
  152. ? $_REQUEST['fields_null_prev']['multi_edit'][$rownumber]
  153. : null;
  154. $multi_edit_auto_increment
  155. = isset($_REQUEST['auto_increment']['multi_edit'][$rownumber])
  156. ? $_REQUEST['auto_increment']['multi_edit'][$rownumber]
  157. : null;
  158. // When a select field is nullified, it's not present in $_REQUEST
  159. // so initialize it; this way, the foreach($multi_edit_colummns) will process it
  160. foreach ($multi_edit_columns_name as $key => $val) {
  161. if (! isset($multi_edit_colummns[$key])) {
  162. $multi_edit_colummns[$key] = '';
  163. }
  164. }
  165. // Iterate in the order of $multi_edit_columns_name,
  166. // not $multi_edit_colummns, to avoid problems
  167. // when inserting multiple entries
  168. foreach ($multi_edit_columns_name as $key => $colummn_name) {
  169. $current_value = $multi_edit_colummns[$key];
  170. // Note: $key is an md5 of the fieldname. The actual fieldname is
  171. // available in $multi_edit_columns_name[$key]
  172. $file_to_insert = new PMA_File();
  173. $file_to_insert->checkTblChangeForm($key, $rownumber);
  174. $possibly_uploaded_val = $file_to_insert->getContent();
  175. if ($file_to_insert->isError()) {
  176. $message .= $file_to_insert->getError();
  177. }
  178. // delete $file_to_insert temporary variable
  179. $file_to_insert->cleanUp();
  180. $current_value = PMA_getCurrentValueForDifferentTypes(
  181. $possibly_uploaded_val, $key, $multi_edit_columns_type,
  182. $current_value, $multi_edit_auto_increment,
  183. $rownumber, $multi_edit_columns_name, $multi_edit_columns_null,
  184. $multi_edit_columns_null_prev, $is_insert,
  185. $using_key, $where_clause, $table
  186. );
  187. $current_value_as_an_array = PMA_getCurrentValueAsAnArrayForMultipleEdit(
  188. $multi_edit_colummns, $multi_edit_columns_name, $multi_edit_funcs,
  189. $multi_edit_salt, $gis_from_text_functions, $current_value,
  190. $gis_from_wkb_functions, $func_optional_param, $func_no_param, $key
  191. );
  192. list($query_values, $query_fields)
  193. = PMA_getQueryValuesForInsertAndUpdateInMultipleEdit(
  194. $multi_edit_columns_name, $multi_edit_columns_null, $current_value,
  195. $multi_edit_columns_prev, $multi_edit_funcs, $is_insert,
  196. $query_values, $query_fields, $current_value_as_an_array,
  197. $value_sets, $key, $multi_edit_columns_null_prev
  198. );
  199. } //end of foreach
  200. if (count($query_values) > 0) {
  201. if ($is_insert) {
  202. $value_sets[] = implode(', ', $query_values);
  203. } else {
  204. // build update query
  205. $query[] = 'UPDATE ' . PMA_Util::backquote($GLOBALS['db'])
  206. . '.' . PMA_Util::backquote($GLOBALS['table'])
  207. . ' SET ' . implode(', ', $query_values)
  208. . ' WHERE ' . $where_clause
  209. . ($_REQUEST['clause_is_unique'] ? '' : ' LIMIT 1');
  210. }
  211. }
  212. } // end foreach ($loop_array as $where_clause)
  213. unset($multi_edit_columns_name, $multi_edit_columns_prev, $multi_edit_funcs,
  214. $multi_edit_columns_type, $multi_edit_columns_null, $func_no_param,
  215. $multi_edit_auto_increment, $current_value_as_an_array, $key, $current_value,
  216. $loop_array, $where_clause, $using_key, $multi_edit_columns_null_prev);
  217. // Builds the sql query
  218. if ($is_insert && count($value_sets) > 0) {
  219. $query = PMA_buildSqlQuery($is_insertignore, $query_fields, $value_sets);
  220. } elseif (empty($query)) {
  221. // No change -> move back to the calling script
  222. //
  223. // Note: logic passes here for inline edit
  224. $message = PMA_Message::success(__('No change'));
  225. $active_page = $goto_include;
  226. include '' . PMA_securePath($goto_include);
  227. exit;
  228. }
  229. unset($multi_edit_colummns, $is_insertignore);
  230. /**
  231. * Executes the sql query and get the result, then move back to the calling
  232. * page
  233. */
  234. list ($url_params, $total_affected_rows, $last_messages, $warning_messages,
  235. $error_messages, $return_to_sql_query)
  236. = PMA_executeSqlQuery($url_params, $query);
  237. if ($is_insert && count($value_sets) > 0) {
  238. $message = PMA_Message::getMessageForInsertedRows($total_affected_rows);
  239. } else {
  240. $message = PMA_Message::getMessageForAffectedRows($total_affected_rows);
  241. }
  242. $message->addMessages($last_messages, '<br />');
  243. if (! empty($warning_messages)) {
  244. $message->addMessages($warning_messages, '<br />');
  245. $message->isError(true);
  246. }
  247. if (! empty($error_messages)) {
  248. $message->addMessages($error_messages);
  249. $message->isError(true);
  250. }
  251. unset(
  252. $error_messages, $warning_messages, $total_affected_rows,
  253. $last_messages, $last_message
  254. );
  255. /**
  256. * The following section only applies to grid editing.
  257. * However, verifying isAjax() is not enough to ensure we are coming from
  258. * grid editing. If we are coming from the Edit or Copy link in Browse mode,
  259. * ajax_page_request is present in the POST parameters.
  260. */
  261. if ($response->isAjax() && ! isset($_POST['ajax_page_request'])) {
  262. /**
  263. * If we are in grid editing, we need to process the relational and
  264. * transformed fields, if they were edited. After that, output the correct
  265. * link/transformed value and exit
  266. *
  267. * Logic taken from libraries/DisplayResults.class.php
  268. */
  269. if (isset($_REQUEST['rel_fields_list']) && $_REQUEST['rel_fields_list'] != '') {
  270. $map = PMA_getForeigners($db, $table, '', 'both');
  271. $relation_fields = array();
  272. parse_str($_REQUEST['rel_fields_list'], $relation_fields);
  273. // loop for each relation cell
  274. foreach ($relation_fields as $cell_index => $curr_rel_field) {
  275. foreach ($curr_rel_field as $relation_field => $relation_field_value) {
  276. $where_comparison = "='" . $relation_field_value . "'";
  277. $dispval = PMA_getDisplayValueForForeignTableColumn(
  278. $where_comparison, $relation_field_value, $map, $relation_field
  279. );
  280. $extra_data['relations'][$cell_index]
  281. = PMA_getLinkForRelationalDisplayField(
  282. $map, $relation_field, $where_comparison,
  283. $dispval, $relation_field_value
  284. );
  285. }
  286. } // end of loop for each relation cell
  287. }
  288. if (isset($_REQUEST['do_transformations'])
  289. && $_REQUEST['do_transformations'] == true
  290. ) {
  291. include_once 'libraries/transformations.lib.php';
  292. //if some posted fields need to be transformed, generate them here.
  293. $mime_map = PMA_getMIME($db, $table);
  294. if ($mime_map === false) {
  295. $mime_map = array();
  296. }
  297. $edited_values = array();
  298. parse_str($_REQUEST['transform_fields_list'], $edited_values);
  299. if (! isset($extra_data)) {
  300. $extra_data = array();
  301. }
  302. foreach ($mime_map as $transformation) {
  303. $file = PMA_securePath($transformation['transformation']);
  304. // if only an underscore in the file name, nothing to transform
  305. if ($file != '_') {
  306. $column_name = $transformation['column_name'];
  307. $extra_data = PMA_transformEditedValues(
  308. $db, $table, $transformation, $edited_values, $file,
  309. $column_name, $extra_data
  310. );
  311. }
  312. } // end of loop for each $mime_map
  313. }
  314. // Need to check the inline edited value can be truncated by MySQL
  315. // without informing while saving
  316. $column_name = $_REQUEST['fields_name']['multi_edit'][0][0];
  317. PMA_verifyWhetherValueCanBeTruncatedAndAppendExtraData(
  318. $db, $table, $column_name, $extra_data
  319. );
  320. /**Get the total row count of the table*/
  321. $extra_data['row_count'] = PMA_Table::countRecords(
  322. $_REQUEST['db'], $_REQUEST['table']
  323. );
  324. $extra_data['sql_query']
  325. = PMA_Util::getMessage($message, $GLOBALS['display_query']);
  326. $response = PMA_Response::getInstance();
  327. $response->isSuccess($message->isSuccess());
  328. $response->addJSON('message', $message);
  329. $response->addJSON($extra_data);
  330. exit;
  331. }
  332. if (! empty($return_to_sql_query)) {
  333. $disp_query = $GLOBALS['sql_query'];
  334. $disp_message = $message;
  335. unset($message);
  336. $GLOBALS['sql_query'] = $return_to_sql_query;
  337. }
  338. $scripts->addFile('tbl_change.js');
  339. $active_page = $goto_include;
  340. /**
  341. * If user asked for "and then Insert another new row" we have to remove
  342. * WHERE clause information so that tbl_change.php does not go back
  343. * to the current record
  344. */
  345. if (isset($_REQUEST['after_insert']) && 'new_insert' == $_REQUEST['after_insert']) {
  346. unset($_REQUEST['where_clause']);
  347. }
  348. /**
  349. * Load target page.
  350. */
  351. require '' . PMA_securePath($goto_include);
  352. exit;
  353. ?>