execute(static::EXECUTE); } /** * @return mixed * @throws Exception */ public function save(): mixed { return $this->execute(static::EXECUTE); } /** * @return mixed * @throws Exception */ public function all(): mixed { return $this->execute(static::FETCH_ALL); } /** * @return mixed * @throws Exception */ public function one(): mixed { return $this->execute(static::FETCH); } /** * @return mixed * @throws Exception */ public function fetchColumn(): mixed { return $this->execute(static::FETCH_COLUMN); } /** * @return mixed * @throws Exception */ public function rowCount(): mixed { return $this->execute(static::ROW_COUNT); } /** * @return mixed * @throws Exception */ public function flush(): mixed { return $this->execute(static::EXECUTE); } /** * @param string $type * @return mixed * @throws Exception */ private function execute(string $type): mixed { $time = microtime(true); $result = $type !== static::EXECUTE ? $this->search($type) : $this->_execute(); $this->logger->debug('Mysql:' . $this->print_r($time)); return $result; } /** * @param $time * @return string */ private function print_r($time): string { return print_r(['time' => microtime(true) - $time, 'sql' => $this->sql, 'param' => $this->params], true); } /** * @return bool|int * @throws Exception */ private function _execute(): bool|int { $pdo = $this->db->getPdo(); try { if (!(($prepare = $pdo->prepare($this->sql)) instanceof PDOStatement)) { throw new Exception($prepare->errorInfo()[2] ?? static::DB_ERROR_MESSAGE); } if ($prepare->execute($this->params) === false) { throw new Exception($prepare->errorInfo()[2] ?? static::DB_ERROR_MESSAGE); } $result = (int)$pdo->lastInsertId(); $prepare->closeCursor(); $this->db->release($pdo, true); return $result == 0 ? true : $result; } catch (\PDOException|\Throwable $throwable) { if (str_contains($throwable->getMessage(), 'MySQL server has gone away')) { Context::remove('master:client'); unset($pdo); return $this->_execute(); } $this->db->release($pdo, true); return $this->logger->addError($this->sql . '. error: ' . $throwable->getMessage(), 'mysql'); } } /** * @param \PDO $pdo * @return PDOStatement * @throws Exception */ private function queryPrev(\PDO $pdo): PDOStatement { try { if (($statement = $pdo->query($this->sql)) === false) { throw new Exception($pdo->errorInfo()[1]); } foreach ($this->params as $key => $param) { $statement->bindValue($key, $param); } return $statement; } catch (\PDOException|\Throwable $throwable) { if (str_contains($throwable->getMessage(), 'MySQL server has gone away')) { unset($pdo); return $this->queryPrev($this->db->getSlaveClient()); } throw new Exception($throwable->getMessage()); } } /** * @param string $type * @return array|int|bool|null * @throws Exception */ private function search(string $type): mixed { $pdo = $this->db->getSlaveClient(); try { if ($type == self::FETCH_ALL) { $data = $this->queryPrev($pdo)->fetchAll(\PDO::FETCH_ASSOC); } else if ($type == self::FETCH) { $data = $this->queryPrev($pdo)->fetch(\PDO::FETCH_ASSOC); } else if ($type == self::FETCH_COLUMN) { $data = $this->queryPrev($pdo)->fetchColumn(\PDO::FETCH_ASSOC); } else if ($type == self::ROW_COUNT) { $data = $this->queryPrev($pdo)->rowCount(); } else { $data = null; } } catch (\Throwable $throwable) { $data = $this->logger->addError($this->sql . '. error: ' . $throwable->getMessage(), 'mysql'); } finally { $this->db->release($pdo, false); return $data; } } /** * @return mixed * @throws Exception */ public function delete(): mixed { return $this->execute(static::EXECUTE); } /** * @return mixed * @throws Exception */ public function exec(): mixed { return $this->execute(static::EXECUTE); } /** * @param array $data * @return $this */ public function bindValues(array $data = []): static { if (!empty($data)) { $this->params = array_merge($this->params, $data); } return $this; } /** * @param $sql * @return $this * @throws Exception */ public function setSql($sql): static { $this->sql = $sql; return $this; } }