OCI8Connection.php 5.73 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
use Doctrine\DBAL\ParameterType;
25

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
    /**
39
     * @var int
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
     * @param string      $username
     * @param string      $password
     * @param string      $db
     * @param string|null $charset
50 51
     * @param int         $sessionMode
     * @param bool        $persistent
Benjamin Morel's avatar
Benjamin Morel committed
52 53
     *
     * @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 = ParameterType::STRING)
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
        if ($name === null) {
            return false;
        }

        $sql    = 'SELECT ' . $name . '.CURRVAL FROM DUAL';
        $stmt   = $this->query($sql);
156
        $result = $stmt->fetchColumn();
157

158
        if ($result === false) {
159 160 161
            throw new OCI8Exception("lastInsertId failed: Query was executed but no result was returned.");
        }

162
        return (int) $result;
163
    }
164

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

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

182 183
        return true;
    }
184 185

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

195
        return true;
196
    }
197 198

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

208
        return true;
209
    }
210

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

221 222
        return $error;
    }
223

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