class Min_Driver extends Min_SQL {
- function insertUpdate($table, $set, $primary) {
- $update = array();
- $where = array();
- foreach ($set as $key => $val) {
- $update[] = "$key = $val";
- if (isset($primary[idf_unescape($key)])) {
- $where[] = "$key = $val";
+ function insertUpdate($table, $rows, $primary) {
+ foreach ($rows as $set) {
+ $update = array();
+ $where = array();
+ foreach ($set as $key => $val) {
+ $update[] = "$key = $val";
+ if (isset($primary[idf_unescape($key)])) {
+ $where[] = "$key = $val";
+ }
+ }
+ //! can use only one query for all rows
+ if (!queries("MERGE " . table($table) . " USING (VALUES(" . implode(", ", $set) . ")) AS source (c" . implode(", c", range(1, count($set))) . ") ON " . implode(" AND ", $where) //! source, c1 - possible conflict
+ . " WHEN MATCHED THEN UPDATE SET " . implode(", ", $update)
+ . " WHEN NOT MATCHED THEN INSERT (" . implode(", ", array_keys($set)) . ") VALUES (" . implode(", ", $set) . ");" // ; is mandatory
+ )) {
+ return false;
}
}
- // can use only one query for all rows with different API
- return queries("MERGE " . table($table) . " USING (VALUES(" . implode(", ", $set) . ")) AS source (c" . implode(", c", range(1, count($set))) . ") ON " . implode(" AND ", $where) //! source, c1 - possible conflict
- . " WHEN MATCHED THEN UPDATE SET " . implode(", ", $update)
- . " WHEN NOT MATCHED THEN INSERT (" . implode(", ", array_keys($set)) . ") VALUES (" . implode(", ", $set) . ");" // ; is mandatory
- );
+ return true;
}
}
return ($set ? parent::insert($table, $set) : queries("INSERT INTO " . table($table) . " ()\nVALUES ()"));
}
- function insertUpdate($table, $set, $primary) {
- foreach ($set as $key => $val) {
- $set[$key] = "$key = VALUES($key)";
+ function insertUpdate($table, $rows, $primary) {
+ $columns = array_keys(reset($rows));
+ $prefix = "INSERT INTO " . table($table) . " (" . implode(", ", $columns) . ") VALUES\n";
+ $values = array();
+ foreach ($columns as $key) {
+ $values[$key] = "$key = VALUES($key)";
}
- $update = implode(", ", $set);
- return queries("INSERT INTO " . table($table) . " SET $update ON DUPLICATE KEY UPDATE $update");
+ $suffix = "\nON DUPLICATE KEY UPDATE " . implode(", ", $values);
+ $values = array();
+ $length = 0;
+ foreach ($rows as $set) {
+ $value = "(" . implode(", ", $set) . ")";
+ if ($values && (strlen($prefix) + $length + strlen($value) + strlen($suffix) > 1e6)) { // 1e6 - default max_allowed_packet
+ if (!queries($prefix . implode(",\n", $values) . $suffix)) {
+ return false;
+ }
+ $values = array();
+ $length = 0;
+ }
+ $values[] = $value;
+ $length += strlen($value) + 2; // 2 - strlen(",\n")
+ }
+ return queries($prefix . implode(",\n", $values) . $suffix);
}
}
class Min_Driver extends Min_SQL {
- function insertUpdate($table, $set, $primary) {
+ function insertUpdate($table, $rows, $primary) {
global $connection;
- $update = array();
- $where = array();
- foreach ($set as $key => $val) {
- $update[] = "$key = $val";
- if (isset($primary[idf_unescape($key)])) {
- $where[] = "$key = $val";
+ foreach ($rows as $set) {
+ $update = array();
+ $where = array();
+ foreach ($set as $key => $val) {
+ $update[] = "$key = $val";
+ if (isset($primary[idf_unescape($key)])) {
+ $where[] = "$key = $val";
+ }
+ }
+ if (!(($where && queries("UPDATE " . table($table) . " SET " . implode(", ", $update) . " WHERE " . implode(" AND ", $where)) && $connection->affected_rows)
+ || queries("INSERT INTO " . table($table) . " (" . implode(", ", array_keys($set)) . ") VALUES (" . implode(", ", $set) . ")")
+ )) {
+ return false;
}
}
- return ($where && queries("UPDATE " . table($table) . " SET " . implode(", ", $update) . " WHERE " . implode(" AND ", $where)) && $connection->affected_rows)
- || queries("INSERT INTO " . table($table) . " (" . implode(", ", array_keys($set)) . ") VALUES (" . implode(", ", $set) . ")")
- ;
+ return true;
}
}
return sdb_request('PutAttributes', $params);
}
- function insertUpdate($table, $set, $primary) {
- return $this->update($table, $set, "WHERE `itemName()` = " . q($set["`itemName()`"]));
+ function insertUpdate($table, $rows, $primary) {
+ //! use one batch request
+ foreach ($rows as $set) {
+ if (!$this->update($table, $set, "WHERE `itemName()` = " . q($set["`itemName()`"]))) {
+ return false;
+ }
+ }
+ return true;
}
}
class Min_Driver extends Min_SQL {
- function insertUpdate($table, $set, $primary) {
- return queries("REPLACE INTO " . table($table) . " (" . implode(", ", array_keys($set)) . ") VALUES (" . implode(", ", $set) . ")");
+ function insertUpdate($table, $rows, $primary) {
+ $values = array();
+ foreach ($rows as $set) {
+ $values[] = "(" . implode(", ", $set) . ")";
+ }
+ return queries("REPLACE INTO " . table($table) . " (" . implode(", ", array_keys(reset($rows))) . ") VALUES\n" . implode(",\n", $values));
}
}
/** Insert or update data in table
* @param string
* @param array
- * @param array columns in keys
+ * @param array of arrays with escaped columns in keys and quoted data in values
* @return bool
*/
- /*abstract*/ function insertUpdate($table, $set, $primary) {
+ /*abstract*/ function insertUpdate($table, $rows, $primary) {
return false;
}
$affected = count($matches[0]);
begin();
$separator = ($_POST["separator"] == "csv" ? "," : ($_POST["separator"] == "tsv" ? "\t" : ";"));
+ $rows = array();
foreach ($matches[0] as $key => $val) {
preg_match_all("~((?>\"[^\"]*\")+|[^$separator]*)$separator~", $val . $separator, $matches2);
if (!$key && !array_diff($matches2[1], $cols)) { //! doesn't work with column names containing ",\n
foreach ($matches2[1] as $i => $col) {
$set[idf_escape($cols[$i])] = ($col == "" && $fields[$cols[$i]]["null"] ? "NULL" : q(str_replace('""', '"', preg_replace('~^"|"$~', '', $col))));
}
- $result = $driver->insertUpdate($TABLE, $set, $primary);
- if (!$result) {
- break;
- }
+ $rows[] = $set;
}
}
+ $result = (!$rows || $driver->insertUpdate($TABLE, $rows, $primary));
if ($result) {
queries("COMMIT");
}
Add button for dropping an index
Display number of selected rows
Disable underlining links
+Improve speed of CSV import
PostgreSQL: Fix handling of nextval() default values
Adminer 3.7.1 (released 2013-06-29):