1 |
<?php
|
2 |
/*
|
3 |
* $Id: Mysql.php 3183 2007-11-18 16:06:37Z romanb $
|
4 |
*
|
5 |
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
6 |
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
7 |
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
8 |
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
9 |
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
10 |
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
11 |
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
12 |
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
13 |
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
14 |
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
15 |
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
16 |
*
|
17 |
* This software consists of voluntary contributions made by many individuals
|
18 |
* and is licensed under the LGPL. For more information, see
|
19 |
* <http://www.phpdoctrine.com>.
|
20 |
*/
|
21 |
Doctrine::autoload('Doctrine_Connection_Common');
|
22 |
/**
|
23 |
* Doctrine_Connection_Mysql
|
24 |
*
|
25 |
* @package Doctrine
|
26 |
* @subpackage Connection
|
27 |
* @license http://www.opensource.org/licenses/lgpl-license.php LGPL
|
28 |
* @author Konsta Vesterinen <kvesteri@cc.hut.fi>
|
29 |
* @author Lukas Smith <smith@pooteeweet.org> (PEAR MDB2 library)
|
30 |
* @version $Revision: 3183 $
|
31 |
* @link www.phpdoctrine.com
|
32 |
* @since 1.0
|
33 |
*/
|
34 |
class Doctrine_Connection_Mysql extends Doctrine_Connection_Common
|
35 |
{
|
36 |
/**
|
37 |
* @var string $driverName the name of this connection driver
|
38 |
*/
|
39 |
protected $driverName = 'Mysql';
|
40 |
|
41 |
/**
|
42 |
* the constructor
|
43 |
*
|
44 |
* @param Doctrine_Manager $manager
|
45 |
* @param PDO|Doctrine_Adapter $adapter database handler
|
46 |
*/
|
47 |
public function __construct(Doctrine_Manager $manager, $adapter)
|
48 |
{
|
49 |
$this->setAttribute(PDO::ATTR_EMULATE_PREPARES, true);
|
50 |
$this->setAttribute(Doctrine::ATTR_DEFAULT_TABLE_TYPE, 'INNODB');
|
51 |
|
52 |
$this->supported = array(
|
53 |
'sequences' => 'emulated',
|
54 |
'indexes' => true,
|
55 |
'affected_rows' => true,
|
56 |
'transactions' => true,
|
57 |
'savepoints' => false,
|
58 |
'summary_functions' => true,
|
59 |
'order_by_text' => true,
|
60 |
'current_id' => 'emulated',
|
61 |
'limit_queries' => true,
|
62 |
'LOBs' => true,
|
63 |
'replace' => true,
|
64 |
'sub_selects' => true,
|
65 |
'auto_increment' => true,
|
66 |
'primary_key' => true,
|
67 |
'result_introspection' => true,
|
68 |
'prepared_statements' => 'emulated',
|
69 |
'identifier_quoting' => true,
|
70 |
'pattern_escaping' => true
|
71 |
);
|
72 |
|
73 |
$this->properties['string_quoting'] = array('start' => "'",
|
74 |
'end' => "'",
|
75 |
'escape' => '\\',
|
76 |
'escape_pattern' => '\\');
|
77 |
|
78 |
$this->properties['identifier_quoting'] = array('start' => '`',
|
79 |
'end' => '`',
|
80 |
'escape' => '`');
|
81 |
|
82 |
$this->properties['sql_comments'] = array(
|
83 |
array('start' => '-- ', 'end' => "\n", 'escape' => false),
|
84 |
array('start' => '#', 'end' => "\n", 'escape' => false),
|
85 |
array('start' => '/*', 'end' => '*/', 'escape' => false),
|
86 |
);
|
87 |
|
88 |
$this->properties['varchar_max_length'] = 255;
|
89 |
|
90 |
parent::__construct($manager, $adapter);
|
91 |
}
|
92 |
|
93 |
/**
|
94 |
* Set the charset on the current connection
|
95 |
*
|
96 |
* @param string charset
|
97 |
*/
|
98 |
public function setCharset($charset)
|
99 |
{
|
100 |
$query = 'SET NAMES ' . $this->quote($charset);
|
101 |
|
102 |
$this->exec($query);
|
103 |
}
|
104 |
|
105 |
/**
|
106 |
* Execute a SQL REPLACE query. A REPLACE query is identical to a INSERT
|
107 |
* query, except that if there is already a row in the table with the same
|
108 |
* key field values, the REPLACE query just updates its values instead of
|
109 |
* inserting a new row.
|
110 |
*
|
111 |
* The REPLACE type of query does not make part of the SQL standards. Since
|
112 |
* practically only MySQL implements it natively, this type of query is
|
113 |
* emulated through this method for other DBMS using standard types of
|
114 |
* queries inside a transaction to assure the atomicity of the operation.
|
115 |
*
|
116 |
* @access public
|
117 |
*
|
118 |
* @param string $table name of the table on which the REPLACE query will
|
119 |
* be executed.
|
120 |
* @param array $fields associative array that describes the fields and the
|
121 |
* values that will be inserted or updated in the specified table. The
|
122 |
* indexes of the array are the names of all the fields of the table. The
|
123 |
* values of the array are also associative arrays that describe the
|
124 |
* values and other properties of the table fields.
|
125 |
*
|
126 |
* Here follows a list of field properties that need to be specified:
|
127 |
*
|
128 |
* value:
|
129 |
* Value to be assigned to the specified field. This value may be
|
130 |
* of specified in database independent type format as this
|
131 |
* function can perform the necessary datatype conversions.
|
132 |
*
|
133 |
* Default:
|
134 |
* this property is required unless the Null property
|
135 |
* is set to 1.
|
136 |
*
|
137 |
* type
|
138 |
* Name of the type of the field. Currently, all types Metabase
|
139 |
* are supported except for clob and blob.
|
140 |
*
|
141 |
* Default: no type conversion
|
142 |
*
|
143 |
* null
|
144 |
* Boolean property that indicates that the value for this field
|
145 |
* should be set to null.
|
146 |
*
|
147 |
* The default value for fields missing in INSERT queries may be
|
148 |
* specified the definition of a table. Often, the default value
|
149 |
* is already null, but since the REPLACE may be emulated using
|
150 |
* an UPDATE query, make sure that all fields of the table are
|
151 |
* listed in this function argument array.
|
152 |
*
|
153 |
* Default: 0
|
154 |
*
|
155 |
* key
|
156 |
* Boolean property that indicates that this field should be
|
157 |
* handled as a primary key or at least as part of the compound
|
158 |
* unique index of the table that will determine the row that will
|
159 |
* updated if it exists or inserted a new row otherwise.
|
160 |
*
|
161 |
* This function will fail if no key field is specified or if the
|
162 |
* value of a key field is set to null because fields that are
|
163 |
* part of unique index they may not be null.
|
164 |
*
|
165 |
* Default: 0
|
166 |
*
|
167 |
* @return integer the number of affected rows
|
168 |
*/
|
169 |
public function replace(Doctrine_Table $table, array $fields, array $keys)
|
170 |
{
|
171 |
$count = count($fields);
|
172 |
$query = $values = '';
|
173 |
$keys = $colnum = 0;
|
174 |
|
175 |
for (reset($fields); $colnum < $count; next($fields), $colnum++) {
|
176 |
$name = key($fields);
|
177 |
|
178 |
if ($colnum > 0) {
|
179 |
$query .= ',';
|
180 |
$values.= ',';
|
181 |
}
|
182 |
|
183 |
$query .= $table->getColumnName($name);
|
184 |
|
185 |
if (isset($fields[$name]['null']) && $fields[$name]['null']) {
|
186 |
$value = 'NULL';
|
187 |
} else {
|
188 |
$type = isset($fields[$name]['type']) ? $fields[$name]['type'] : null;
|
189 |
$value = $this->quote($fields[$name]['value'], $type);
|
190 |
}
|
191 |
|
192 |
$values .= $value;
|
193 |
|
194 |
if (isset($fields[$name]['key']) && $fields[$name]['key']) {
|
195 |
if ($value === 'NULL') {
|
196 |
throw new Doctrine_Connection_Mysql_Exception('key value '.$name.' may not be NULL');
|
197 |
}
|
198 |
$keys++;
|
199 |
}
|
200 |
}
|
201 |
|
202 |
if ($keys == 0) {
|
203 |
throw new Doctrine_Connection_Mysql_Exception('not specified which fields are keys');
|
204 |
}
|
205 |
$query = 'REPLACE INTO ' . $table->getTableName() . ' (' . $query . ') VALUES (' . $values . ')';
|
206 |
|
207 |
return $this->exec($query);
|
208 |
}
|
209 |
}
|