OCI8Connection.php 5.75 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
<?php
/*
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 *
 * This software consists of voluntary contributions made by many individuals
Benjamin Eberlei's avatar
Benjamin Eberlei committed
16
 * and is licensed under the MIT license. For more information, see
17 18 19 20 21
 * <http://www.doctrine-project.org>.
 */

namespace Doctrine\DBAL\Driver\OCI8;

Steve Müller's avatar
Steve Müller committed
22
use Doctrine\DBAL\Driver\Connection;
23
use Doctrine\DBAL\Driver\ServerInfoAwareConnection;
24

25 26 27 28 29
/**
 * OCI8 implementation of the Connection interface.
 *
 * @since 2.0
 */
30
class OCI8Connection implements Connection, ServerInfoAwareConnection
31
{
32 33 34 35
    /**
     * @var resource
     */
    protected $dbh;
36

37
    /**
Benjamin Morel's avatar
Benjamin Morel committed
38
     * @var integer
39
     */
40
    protected $executeMode = OCI_COMMIT_ON_SUCCESS;
41

42
    /**
Benjamin Morel's avatar
Benjamin Morel committed
43
     * Creates a Connection to an Oracle Database using oci8 extension.
44
     *
Benjamin Morel's avatar
Benjamin Morel committed
45 46 47 48 49 50 51 52
     * @param string      $username
     * @param string      $password
     * @param string      $db
     * @param string|null $charset
     * @param integer     $sessionMode
     * @param boolean     $persistent
     *
     * @throws OCI8Exception
53
     */
54
    public function __construct($username, $password, $db, $charset = null, $sessionMode = OCI_DEFAULT, $persistent = false)
55
    {
56 57 58 59
        if (!defined('OCI_NO_AUTO_COMMIT')) {
            define('OCI_NO_AUTO_COMMIT', 0);
        }

60
        $this->dbh = $persistent
61 62
            ? @oci_pconnect($username, $password, $db, $charset, $sessionMode)
            : @oci_connect($username, $password, $db, $charset, $sessionMode);
63

64
        if ( ! $this->dbh) {
65
            throw OCI8Exception::fromErrorInfo(oci_error());
66
        }
67
    }
68

69 70
    /**
     * {@inheritdoc}
71 72 73
     *
     * @throws \UnexpectedValueException if the version string returned by the database server
     *                                   does not contain a parsable version number.
74 75 76 77
     */
    public function getServerVersion()
    {
        if ( ! preg_match('/\s+(\d+\.\d+\.\d+\.\d+\.\d+)\s+/', oci_server_version($this->dbh), $version)) {
78 79 80 81 82 83 84
            throw new \UnexpectedValueException(
                sprintf(
                    'Unexpected database version string "%s". Cannot parse an appropriate version number from it. ' .
                    'Please report this database version string to the Doctrine team.',
                    oci_server_version($this->dbh)
                )
            );
85 86 87 88 89 90 91 92 93 94 95 96 97
        }

        return $version[1];
    }

    /**
     * {@inheritdoc}
     */
    public function requiresQueryForServerVersion()
    {
        return false;
    }

98
    /**
Benjamin Morel's avatar
Benjamin Morel committed
99
     * {@inheritdoc}
100
     */
101 102
    public function prepare($prepareString)
    {
103
        return new OCI8Statement($this->dbh, $prepareString, $this);
104
    }
105 106

    /**
Benjamin Morel's avatar
Benjamin Morel committed
107
     * {@inheritdoc}
108
     */
109 110 111 112 113 114 115
    public function query()
    {
        $args = func_get_args();
        $sql = $args[0];
        //$fetchMode = $args[1];
        $stmt = $this->prepare($sql);
        $stmt->execute();
Benjamin Morel's avatar
Benjamin Morel committed
116

117 118
        return $stmt;
    }
119 120

    /**
Benjamin Morel's avatar
Benjamin Morel committed
121
     * {@inheritdoc}
122
     */
123
    public function quote($value, $type=\PDO::PARAM_STR)
124
    {
125 126 127 128
        if (is_int($value) || is_float($value)) {
            return $value;
        }
        $value = str_replace("'", "''", $value);
Benjamin Morel's avatar
Benjamin Morel committed
129

130
        return "'" . addcslashes($value, "\000\n\r\\\032") . "'";
131
    }
132 133

    /**
Benjamin Morel's avatar
Benjamin Morel committed
134
     * {@inheritdoc}
135
     */
136 137 138 139
    public function exec($statement)
    {
        $stmt = $this->prepare($statement);
        $stmt->execute();
Benjamin Morel's avatar
Benjamin Morel committed
140

141 142
        return $stmt->rowCount();
    }
143

144
    /**
Benjamin Morel's avatar
Benjamin Morel committed
145
     * {@inheritdoc}
146
     */
147 148
    public function lastInsertId($name = null)
    {
149 150 151 152 153 154 155 156 157 158 159 160 161
        if ($name === null) {
            return false;
        }

        $sql    = 'SELECT ' . $name . '.CURRVAL FROM DUAL';
        $stmt   = $this->query($sql);
        $result = $stmt->fetch(\PDO::FETCH_ASSOC);

        if ($result === false || !isset($result['CURRVAL'])) {
            throw new OCI8Exception("lastInsertId failed: Query was executed but no result was returned.");
        }

        return (int) $result['CURRVAL'];
162
    }
163

164
    /**
Benjamin Morel's avatar
Benjamin Morel committed
165 166 167
     * Returns the current execution mode.
     *
     * @return integer
168 169 170
     */
    public function getExecuteMode()
    {
171
        return $this->executeMode;
172 173
    }

174
    /**
Benjamin Morel's avatar
Benjamin Morel committed
175
     * {@inheritdoc}
176
     */
177 178
    public function beginTransaction()
    {
179
        $this->executeMode = OCI_NO_AUTO_COMMIT;
Benjamin Morel's avatar
Benjamin Morel committed
180

181 182
        return true;
    }
183 184

    /**
Benjamin Morel's avatar
Benjamin Morel committed
185
     * {@inheritdoc}
186
     */
187 188
    public function commit()
    {
189
        if (!oci_commit($this->dbh)) {
190 191
            throw OCI8Exception::fromErrorInfo($this->errorInfo());
        }
192
        $this->executeMode = OCI_COMMIT_ON_SUCCESS;
Benjamin Morel's avatar
Benjamin Morel committed
193

194
        return true;
195
    }
196 197

    /**
Benjamin Morel's avatar
Benjamin Morel committed
198
     * {@inheritdoc}
199
     */
200 201
    public function rollBack()
    {
202
        if (!oci_rollback($this->dbh)) {
203 204
            throw OCI8Exception::fromErrorInfo($this->errorInfo());
        }
205
        $this->executeMode = OCI_COMMIT_ON_SUCCESS;
Benjamin Morel's avatar
Benjamin Morel committed
206

207
        return true;
208
    }
209

Benjamin Morel's avatar
Benjamin Morel committed
210 211 212
    /**
     * {@inheritdoc}
     */
213 214
    public function errorCode()
    {
215
        $error = oci_error($this->dbh);
216 217 218
        if ($error !== false) {
            $error = $error['code'];
        }
Benjamin Morel's avatar
Benjamin Morel committed
219

220 221
        return $error;
    }
222

Benjamin Morel's avatar
Benjamin Morel committed
223 224 225
    /**
     * {@inheritdoc}
     */
226 227
    public function errorInfo()
    {
228
        return oci_error($this->dbh);
229
    }
230
}