OCI8Connection.php 5.3 KB
Newer Older
1 2 3 4
<?php

namespace Doctrine\DBAL\Driver\OCI8;

Steve Müller's avatar
Steve Müller committed
5
use Doctrine\DBAL\Driver\Connection;
6
use Doctrine\DBAL\Driver\ServerInfoAwareConnection;
7
use Doctrine\DBAL\ParameterType;
8
use UnexpectedValueException;
9

10 11 12 13 14 15 16 17 18 19 20 21 22
use function addcslashes;
use function func_get_args;
use function is_float;
use function is_int;
use function oci_commit;
use function oci_connect;
use function oci_error;
use function oci_pconnect;
use function oci_rollback;
use function oci_server_version;
use function preg_match;
use function sprintf;
use function str_replace;
23

Grégoire Paris's avatar
Grégoire Paris committed
24 25
use const OCI_COMMIT_ON_SUCCESS;
use const OCI_NO_AUTO_COMMIT;
26

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

35
    /** @var int */
36
    protected $executeMode = OCI_COMMIT_ON_SUCCESS;
37

38
    /**
Benjamin Morel's avatar
Benjamin Morel committed
39
     * Creates a Connection to an Oracle Database using oci8 extension.
40
     *
Sergei Morozov's avatar
Sergei Morozov committed
41 42 43 44 45 46
     * @param string $username
     * @param string $password
     * @param string $db
     * @param string $charset
     * @param int    $sessionMode
     * @param bool   $persistent
Benjamin Morel's avatar
Benjamin Morel committed
47 48
     *
     * @throws OCI8Exception
49
     */
Sergei Morozov's avatar
Sergei Morozov committed
50 51 52 53 54
    public function __construct(
        $username,
        $password,
        $db,
        $charset = '',
55
        $sessionMode = OCI_NO_AUTO_COMMIT,
Sergei Morozov's avatar
Sergei Morozov committed
56 57
        $persistent = false
    ) {
Sergei Morozov's avatar
Sergei Morozov committed
58
        $dbh = $persistent
59 60
            ? @oci_pconnect($username, $password, $db, $charset, $sessionMode)
            : @oci_connect($username, $password, $db, $charset, $sessionMode);
61

Sergei Morozov's avatar
Sergei Morozov committed
62
        if ($dbh === false) {
63
            throw OCI8Exception::fromErrorInfo(oci_error());
64
        }
Sergei Morozov's avatar
Sergei Morozov committed
65 66

        $this->dbh = $dbh;
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
     */
    public function getServerVersion()
    {
Sergei Morozov's avatar
Sergei Morozov committed
77 78 79 80 81 82 83
        $version = oci_server_version($this->dbh);

        if ($version === false) {
            throw OCI8Exception::fromErrorInfo(oci_error($this->dbh));
        }

        if (! preg_match('/\s+(\d+\.\d+\.\d+\.\d+\.\d+)\s+/', $version, $matches)) {
84
            throw new UnexpectedValueException(
85 86 87
                sprintf(
                    'Unexpected database version string "%s". Cannot parse an appropriate version number from it. ' .
                    'Please report this database version string to the Doctrine team.',
Sergei Morozov's avatar
Sergei Morozov committed
88
                    $version
89 90
                )
            );
91 92
        }

Sergei Morozov's avatar
Sergei Morozov committed
93
        return $matches[1];
94 95 96 97 98 99 100 101 102 103
    }

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

104
    /**
Benjamin Morel's avatar
Benjamin Morel committed
105
     * {@inheritdoc}
106
     */
107
    public function prepare($sql)
108
    {
109
        return new OCI8Statement($this->dbh, $sql, $this);
110
    }
111 112

    /**
Benjamin Morel's avatar
Benjamin Morel committed
113
     * {@inheritdoc}
114
     */
115 116 117
    public function query()
    {
        $args = func_get_args();
118
        $sql  = $args[0];
119 120 121
        //$fetchMode = $args[1];
        $stmt = $this->prepare($sql);
        $stmt->execute();
Benjamin Morel's avatar
Benjamin Morel committed
122

123 124
        return $stmt;
    }
125 126

    /**
Benjamin Morel's avatar
Benjamin Morel committed
127
     * {@inheritdoc}
128
     */
129
    public function quote($value, $type = ParameterType::STRING)
130
    {
131 132 133
        if (is_int($value) || is_float($value)) {
            return $value;
        }
Grégoire Paris's avatar
Grégoire Paris committed
134

135
        $value = str_replace("'", "''", $value);
Benjamin Morel's avatar
Benjamin Morel committed
136

137
        return "'" . addcslashes($value, "\000\n\r\\\032") . "'";
138
    }
139 140

    /**
Benjamin Morel's avatar
Benjamin Morel committed
141
     * {@inheritdoc}
142
     */
143
    public function exec($sql)
144
    {
145
        $stmt = $this->prepare($sql);
146
        $stmt->execute();
Benjamin Morel's avatar
Benjamin Morel committed
147

148 149
        return $stmt->rowCount();
    }
150

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

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

166
        if ($result === false) {
167
            throw new OCI8Exception('lastInsertId failed: Query was executed but no result was returned.');
168 169
        }

170
        return (int) $result;
171
    }
172

173
    /**
Benjamin Morel's avatar
Benjamin Morel committed
174 175
     * Returns the current execution mode.
     *
176
     * @return int
177 178 179
     */
    public function getExecuteMode()
    {
180
        return $this->executeMode;
181 182
    }

183
    /**
Benjamin Morel's avatar
Benjamin Morel committed
184
     * {@inheritdoc}
185
     */
186 187
    public function beginTransaction()
    {
188
        $this->executeMode = OCI_NO_AUTO_COMMIT;
Benjamin Morel's avatar
Benjamin Morel committed
189

190 191
        return true;
    }
192 193

    /**
Benjamin Morel's avatar
Benjamin Morel committed
194
     * {@inheritdoc}
195
     */
196 197
    public function commit()
    {
198
        if (! oci_commit($this->dbh)) {
199 200
            throw OCI8Exception::fromErrorInfo($this->errorInfo());
        }
Grégoire Paris's avatar
Grégoire Paris committed
201

202
        $this->executeMode = OCI_COMMIT_ON_SUCCESS;
Benjamin Morel's avatar
Benjamin Morel committed
203

204
        return true;
205
    }
206 207

    /**
Benjamin Morel's avatar
Benjamin Morel committed
208
     * {@inheritdoc}
209
     */
210 211
    public function rollBack()
    {
212
        if (! oci_rollback($this->dbh)) {
213 214
            throw OCI8Exception::fromErrorInfo($this->errorInfo());
        }
Grégoire Paris's avatar
Grégoire Paris committed
215

216
        $this->executeMode = OCI_COMMIT_ON_SUCCESS;
Benjamin Morel's avatar
Benjamin Morel committed
217

218
        return true;
219
    }
220

Benjamin Morel's avatar
Benjamin Morel committed
221 222 223
    /**
     * {@inheritdoc}
     */
224 225
    public function errorCode()
    {
226
        $error = oci_error($this->dbh);
227

228
        if ($error !== false) {
229
            return $error['code'];
230
        }
Benjamin Morel's avatar
Benjamin Morel committed
231

232
        return null;
233
    }
234

Benjamin Morel's avatar
Benjamin Morel committed
235 236 237
    /**
     * {@inheritdoc}
     */
238 239
    public function errorInfo()
    {
Sergei Morozov's avatar
Sergei Morozov committed
240 241 242 243 244 245 246
        $error = oci_error($this->dbh);

        if ($error === false) {
            return [];
        }

        return $error;
247
    }
248
}