OCI8Connection.php 5.84 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
use Doctrine\DBAL\Platforms\OraclePlatform;

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

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

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

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

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

70 71
    /**
     * {@inheritdoc}
72 73 74
     *
     * @throws \UnexpectedValueException if the version string returned by the database server
     *                                   does not contain a parsable version number.
75 76 77 78
     */
    public function getServerVersion()
    {
        if ( ! preg_match('/\s+(\d+\.\d+\.\d+\.\d+\.\d+)\s+/', oci_server_version($this->dbh), $version)) {
79 80 81 82 83 84 85
            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)
                )
            );
86 87 88 89 90 91 92 93 94 95 96 97 98
        }

        return $version[1];
    }

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

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

    /**
Benjamin Morel's avatar
Benjamin Morel committed
108
     * {@inheritdoc}
109
     */
110 111 112 113 114 115 116
    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
117

118 119
        return $stmt;
    }
120 121

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

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

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

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

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

        OraclePlatform::assertValidIdentifier($name);

        $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'];
165
    }
166

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

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

184 185
        return true;
    }
186 187

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

197
        return true;
198
    }
199 200

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

210
        return true;
211
    }
212

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

223 224
        return $error;
    }
225

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