Browse Source
Merge pull request #9444 from nextcloud/techdep/noid/appframework_mapper_to_qb
Merge pull request #9444 from nextcloud/techdep/noid/appframework_mapper_to_qb
Add a QueryBuilder based Mapperpull/8760/head
committed by
GitHub
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 290 additions and 2 deletions
-
1lib/composer/composer/autoload_classmap.php
-
1lib/composer/composer/autoload_static.php
-
3lib/private/Authentication/Token/DefaultTokenMapper.php
-
15lib/public/AppFramework/Db/Mapper.php
-
272lib/public/AppFramework/Db/QBMapper.php
@ -0,0 +1,272 @@ |
|||
<?php |
|||
declare(strict_types=1); |
|||
/** |
|||
* @copyright 2018, Roeland Jago Douma <roeland@famdouma.nl> |
|||
* |
|||
* @author Roeland Jago Douma <roeland@famdouma.nl> |
|||
* |
|||
* @license GNU AGPL version 3 or any later version |
|||
* |
|||
* This program is free software: you can redistribute it and/or modify |
|||
* it under the terms of the GNU Affero General Public License as |
|||
* published by the Free Software Foundation, either version 3 of the |
|||
* License, or (at your option) any later version. |
|||
* |
|||
* This program is distributed in the hope that it will be useful, |
|||
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
|||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|||
* GNU Affero General Public License for more details. |
|||
* |
|||
* You should have received a copy of the GNU Affero General Public License |
|||
* along with this program. If not, see <http://www.gnu.org/licenses/>. |
|||
* |
|||
*/ |
|||
|
|||
namespace OCP\AppFramework\Db; |
|||
|
|||
use OCP\DB\QueryBuilder\IQueryBuilder; |
|||
use OCP\IDBConnection; |
|||
|
|||
/** |
|||
* Simple parent class for inheriting your data access layer from. This class |
|||
* may be subject to change in the future |
|||
* |
|||
* @since 14.0.0 |
|||
*/ |
|||
abstract class QBMapper { |
|||
|
|||
/** @var string */ |
|||
protected $tableName; |
|||
|
|||
/** @var string */ |
|||
protected $entityClass; |
|||
|
|||
/** @var IDBConnection */ |
|||
protected $db; |
|||
|
|||
/** |
|||
* @param IDBConnection $db Instance of the Db abstraction layer |
|||
* @param string $tableName the name of the table. set this to allow entity |
|||
* @param string $entityClass the name of the entity that the sql should be |
|||
* mapped to queries without using sql |
|||
* @since 14.0.0 |
|||
*/ |
|||
public function __construct(IDBConnection $db, string $tableName, string $entityClass=null){ |
|||
$this->db = $db; |
|||
$this->tableName = $tableName; |
|||
|
|||
// if not given set the entity name to the class without the mapper part
|
|||
// cache it here for later use since reflection is slow
|
|||
if($entityClass === null) { |
|||
$this->entityClass = str_replace('Mapper', '', \get_class($this)); |
|||
} else { |
|||
$this->entityClass = $entityClass; |
|||
} |
|||
} |
|||
|
|||
|
|||
/** |
|||
* @return string the table name |
|||
* @since 14.0.0 |
|||
*/ |
|||
public function getTableName(): string { |
|||
return $this->tableName; |
|||
} |
|||
|
|||
|
|||
/** |
|||
* Deletes an entity from the table |
|||
* @param Entity $entity the entity that should be deleted |
|||
* @return Entity the deleted entity |
|||
* @since 14.0.0 |
|||
*/ |
|||
public function delete(Entity $entity): Entity { |
|||
$qb = $this->db->getQueryBuilder(); |
|||
|
|||
$qb->delete($this->tableName) |
|||
->where( |
|||
$qb->expr()->eq('id', $qb->createNamedParameter($entity->getId())) |
|||
); |
|||
$qb->execute(); |
|||
return $entity; |
|||
} |
|||
|
|||
|
|||
/** |
|||
* Creates a new entry in the db from an entity |
|||
* @param Entity $entity the entity that should be created |
|||
* @return Entity the saved entity with the set id |
|||
* @since 14.0.0 |
|||
* @suppress SqlInjectionChecker |
|||
*/ |
|||
public function insert(Entity $entity): Entity { |
|||
// get updated fields to save, fields have to be set using a setter to
|
|||
// be saved
|
|||
$properties = $entity->getUpdatedFields(); |
|||
|
|||
$qb = $this->db->getQueryBuilder(); |
|||
$qb->insert($this->tableName); |
|||
|
|||
// build the fields
|
|||
foreach($properties as $property => $updated) { |
|||
$column = $entity->propertyToColumn($property); |
|||
$getter = 'get' . ucfirst($property); |
|||
$value = $entity->$getter(); |
|||
|
|||
$qb->setValue($column, $qb->createNamedParameter($value)); |
|||
} |
|||
|
|||
$qb->execute(); |
|||
|
|||
$entity->setId((int) $qb->getLastInsertId()); |
|||
|
|||
return $entity; |
|||
} |
|||
|
|||
|
|||
|
|||
/** |
|||
* Updates an entry in the db from an entity |
|||
* @throws \InvalidArgumentException if entity has no id |
|||
* @param Entity $entity the entity that should be created |
|||
* @return Entity the saved entity with the set id |
|||
* @since 14.0.0 |
|||
* @suppress SqlInjectionChecker |
|||
*/ |
|||
public function update(Entity $entity): Entity { |
|||
// if entity wasn't changed it makes no sense to run a db query
|
|||
$properties = $entity->getUpdatedFields(); |
|||
if(\count($properties) === 0) { |
|||
return $entity; |
|||
} |
|||
|
|||
// entity needs an id
|
|||
$id = $entity->getId(); |
|||
if($id === null){ |
|||
throw new \InvalidArgumentException( |
|||
'Entity which should be updated has no id'); |
|||
} |
|||
|
|||
// get updated fields to save, fields have to be set using a setter to
|
|||
// be saved
|
|||
// do not update the id field
|
|||
unset($properties['id']); |
|||
|
|||
$qb = $this->db->getQueryBuilder(); |
|||
$qb->update($this->tableName); |
|||
|
|||
// build the fields
|
|||
foreach($properties as $property => $updated) { |
|||
$column = $entity->propertyToColumn($property); |
|||
$getter = 'get' . ucfirst($property); |
|||
$value = $entity->$getter(); |
|||
|
|||
$qb->set($column, $qb->createNamedParameter($value)); |
|||
} |
|||
|
|||
$qb->where( |
|||
$qb->expr()->eq('id', $qb->createNamedParameter($id)) |
|||
); |
|||
$qb->execute(); |
|||
|
|||
return $entity; |
|||
} |
|||
|
|||
/** |
|||
* Returns an db result and throws exceptions when there are more or less |
|||
* results |
|||
* |
|||
* @see findEntity |
|||
* |
|||
* @param IQueryBuilder $query |
|||
* @throws DoesNotExistException if the item does not exist |
|||
* @throws MultipleObjectsReturnedException if more than one item exist |
|||
* @return array the result as row |
|||
* @since 14.0.0 |
|||
*/ |
|||
protected function findOneQuery(IQueryBuilder $query): array { |
|||
$cursor = $query->execute(); |
|||
|
|||
$row = $cursor->fetch(); |
|||
if($row === false) { |
|||
$cursor->closeCursor(); |
|||
$msg = $this->buildDebugMessage( |
|||
'Did expect one result but found none when executing', $query |
|||
); |
|||
throw new DoesNotExistException($msg); |
|||
} |
|||
|
|||
$row2 = $cursor->fetch(); |
|||
$cursor->closeCursor(); |
|||
if($row2 !== false ) { |
|||
$msg = $this->buildDebugMessage( |
|||
'Did not expect more than one result when executing', $query |
|||
); |
|||
throw new MultipleObjectsReturnedException($msg); |
|||
} |
|||
|
|||
return $row; |
|||
} |
|||
|
|||
/** |
|||
* @param string $msg |
|||
* @param IQueryBuilder $sql |
|||
* @return string |
|||
* @since 14.0.0 |
|||
*/ |
|||
private function buildDebugMessage(string $msg, IQueryBuilder $sql): string { |
|||
return $msg . |
|||
': query "' . $sql->getSQL() . '"; '; |
|||
} |
|||
|
|||
|
|||
/** |
|||
* Creates an entity from a row. Automatically determines the entity class |
|||
* from the current mapper name (MyEntityMapper -> MyEntity) |
|||
* |
|||
* @param array $row the row which should be converted to an entity |
|||
* @return Entity the entity |
|||
* @since 14.0.0 |
|||
*/ |
|||
protected function mapRowToEntity(array $row): Entity { |
|||
return \call_user_func($this->entityClass .'::fromRow', $row); |
|||
} |
|||
|
|||
|
|||
/** |
|||
* Runs a sql query and returns an array of entities |
|||
* |
|||
* @param IQueryBuilder $query |
|||
* @return Entity[] all fetched entities |
|||
* @since 14.0.0 |
|||
*/ |
|||
protected function findEntities(IQueryBuilder $query): array { |
|||
$cursor = $query->execute(); |
|||
|
|||
$entities = []; |
|||
|
|||
while($row = $cursor->fetch()){ |
|||
$entities[] = $this->mapRowToEntity($row); |
|||
} |
|||
|
|||
$cursor->closeCursor(); |
|||
|
|||
return $entities; |
|||
} |
|||
|
|||
|
|||
/** |
|||
* Returns an db result and throws exceptions when there are more or less |
|||
* results |
|||
* |
|||
* @param IQueryBuilder $query |
|||
* @throws DoesNotExistException if the item does not exist |
|||
* @throws MultipleObjectsReturnedException if more than one item exist |
|||
* @return Entity the entity |
|||
* @since 14.0.0 |
|||
*/ |
|||
protected function findEntity(IQueryBuilder $query): Entity { |
|||
return $this->mapRowToEntity($this->findOneQuery($query)); |
|||
} |
|||
|
|||
} |
|||
Write
Preview
Loading…
Cancel
Save
Reference in new issue