mirror of
https://github.com/nextcloud/documentation.git
synced 2026-01-03 02:09:45 +07:00
This moves lots of pages around. The high-level changes are * Better main sections, so it's more *general*, *into*, *basics* and *details* * Move more general topics to a *Basic* section, which are not app-specific * Remove app docs to the stuff that is likely used, anything else goes into "Digging deeper" * Move general guides into a prologue * Try to *compress*/combine some pages with similar content * Try to have better consistencs on level ob abstraction across pages * Split app development and maintenance pages into two sections * Integrate bugtracker info into prologue * Integrate Android pages into client APIs section Signed-off-by: Christoph Wurst <christoph@winzerhof-wurst.at>
257 lines
7.1 KiB
ReStructuredText
257 lines
7.1 KiB
ReStructuredText
===============
|
|
Database access
|
|
===============
|
|
|
|
.. sectionauthor:: Bernhard Posselt <dev@bernhard-posselt.com>
|
|
|
|
The basic way to run a database query is to use the database connection provided by **OCP\\IDBConnection**.
|
|
|
|
Inside your database layer class you can now start running queries like:
|
|
|
|
.. code-block:: php
|
|
|
|
<?php
|
|
// db/authordao.php
|
|
|
|
namespace OCA\MyApp\Db;
|
|
|
|
use OCP\DB\QueryBuilder\IQueryBuilder;
|
|
use OCP\IDBConnection;
|
|
|
|
class AuthorDAO {
|
|
|
|
private $db;
|
|
|
|
public function __construct(IDBConnection $db) {
|
|
$this->db = $db;
|
|
}
|
|
|
|
public function find(int $id) {
|
|
$qb = $this->db->getQueryBuilder();
|
|
|
|
$qb->select('*')
|
|
->from('myapp_authors')
|
|
->where(
|
|
$qb->expr()->eq('id', $qb->createNamedParameter($id, IQueryBuilder::PARAM_INT))
|
|
);
|
|
|
|
$cursor = $qb->execute();
|
|
$row = $cursor->fetch();
|
|
$cursor->closeCursor();
|
|
|
|
return $row;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
Mappers
|
|
-------
|
|
|
|
The aforementioned example is the most basic way to write a simple database query but the more queries amass, the more code has to be written and the harder it will become to maintain it.
|
|
|
|
To generalize and simplify the problem, split code into resources and create an **Entity** and a **Mapper** class for it. The mapper class provides a way to run SQL queries and maps the result onto the related entities.
|
|
|
|
|
|
To create a mapper, inherit from the mapper base class and call the parent constructor with the following parameters:
|
|
|
|
* Database connection
|
|
* Table name
|
|
* **Optional**: Entity class name, defaults to \\OCA\\MyApp\\Db\\Author in the example below
|
|
|
|
.. code-block:: php
|
|
|
|
<?php
|
|
// db/authormapper.php
|
|
|
|
namespace OCA\MyApp\Db;
|
|
|
|
use OCP\DB\QueryBuilder\IQueryBuilder;
|
|
use OCP\IDBConnection;
|
|
use OCP\AppFramework\Db\QBMapper;
|
|
|
|
class AuthorMapper extends QBMapper {
|
|
|
|
public function __construct(IDBConnection $db) {
|
|
parent::__construct($db, 'myapp_authors');
|
|
}
|
|
|
|
|
|
/**
|
|
* @throws \OCP\AppFramework\Db\DoesNotExistException if not found
|
|
* @throws \OCP\AppFramework\Db\MultipleObjectsReturnedException if more than one result
|
|
*/
|
|
public function find(int $id) {
|
|
$qb = $this->db->getQueryBuilder();
|
|
|
|
$qb->select('*')
|
|
->from('myapp_authors')
|
|
->where(
|
|
$qb->expr()->eq('id', $qb->createNamedParameter($id, IQueryBuilder::PARAM_INT))
|
|
);
|
|
|
|
return $this->findEntity($qb);
|
|
}
|
|
|
|
|
|
public function findAll($limit=null, $offset=null) {
|
|
$qb = $this->db->getQueryBuilder();
|
|
|
|
$qb->select('*')
|
|
->from('myapp_authors')
|
|
->setMaxResults($limit)
|
|
->setFirstResult($offset);
|
|
|
|
return $this->findEntities($sql);
|
|
}
|
|
|
|
|
|
public function authorNameCount($name) {
|
|
$qb = $this->db->getQueryBuilder();
|
|
|
|
$qb->selectAlias($qb->createFunction('COUNT(*)'), 'count')
|
|
->from('myapp_authors')
|
|
->where(
|
|
$qb->expr()->eq('name', $qb->createNamedParameter($name, IQueryBuilder::PARAM_STR))
|
|
);
|
|
|
|
$cursor = $qb->execute();
|
|
$row = $cursor->fetch();
|
|
$cursor->closeCursor();
|
|
|
|
return $row['count'];
|
|
}
|
|
|
|
}
|
|
|
|
.. note:: The cursor is closed automatically for all **INSERT**, **DELETE**, **UPDATE** queries and when calling the methods **findOneQuery**, **findEntities**, **findEntity**, **delete**, **insert** and **update**. For custom calls using execute you should always close the cursor after you are done with the fetching to prevent database lock problems on SQLite
|
|
|
|
Every mapper also implements default methods for deleting and updating an entity based on its id::
|
|
|
|
$authorMapper->delete($entity);
|
|
|
|
or::
|
|
|
|
$authorMapper->update($entity);
|
|
|
|
|
|
|
|
Entities
|
|
--------
|
|
|
|
Entities are data objects that carry all the table's information for one row. Every Entity has an **id** field by default that is set to the integer type. Table rows are mapped from lower case and underscore separated names to *lowerCamelCase* attributes:
|
|
|
|
* **Table column name**: phone_number
|
|
* **Property name**: phoneNumber
|
|
|
|
.. code-block:: php
|
|
|
|
<?php
|
|
// db/author.php
|
|
namespace OCA\MyApp\Db;
|
|
|
|
use OCP\AppFramework\Db\Entity;
|
|
|
|
class Author extends Entity {
|
|
|
|
protected $stars;
|
|
protected $name;
|
|
protected $phoneNumber;
|
|
|
|
public function __construct() {
|
|
// add types in constructor
|
|
$this->addType('stars', 'integer');
|
|
}
|
|
}
|
|
|
|
Types
|
|
^^^^^
|
|
|
|
The following properties should be annotated by types, to not only assure that the types are converted correctly for storing them in the database (e.g. PHP casts false to the empty string which fails on PostgreSQL) but also for casting them when they are retrieved from the database.
|
|
|
|
The following types can be added for a field:
|
|
|
|
* integer
|
|
* float
|
|
* boolean
|
|
|
|
Accessing attributes
|
|
^^^^^^^^^^^^^^^^^^^^
|
|
|
|
Since all attributes should be protected, getters and setters are automatically generated for you:
|
|
|
|
|
|
.. code-block:: php
|
|
|
|
<?php
|
|
// db/author.php
|
|
namespace OCA\MyApp\Db;
|
|
|
|
use OCP\AppFramework\Db\Entity;
|
|
|
|
class Author extends Entity {
|
|
protected $stars;
|
|
protected $name;
|
|
protected $phoneNumber;
|
|
}
|
|
|
|
$author = new Author();
|
|
$author->setId(3);
|
|
$author->getPhoneNumber() // null
|
|
|
|
Custom attribute to database column mapping
|
|
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
|
|
|
By default each attribute will be mapped to a database column by a certain convention, e.g. **phoneNumber**
|
|
will be mapped to the column **phone_number** and vice versa. Sometimes it is needed though to map attributes to
|
|
different columns because of backwards compatibility. To define a custom
|
|
mapping, simply override the **columnToProperty** and **propertyToColumn** methods of the entity in question:
|
|
|
|
.. code-block:: php
|
|
|
|
|
|
<?php
|
|
// db/author.php
|
|
namespace OCA\MyApp\Db;
|
|
|
|
use OCP\AppFramework\Db\Entity;
|
|
|
|
class Author extends Entity {
|
|
protected $stars;
|
|
protected $name;
|
|
protected $phoneNumber;
|
|
|
|
// map attribute phoneNumber to the database column phonenumber
|
|
public function columnToProperty($column) {
|
|
if ($column === 'phonenumber') {
|
|
return 'phoneNumber';
|
|
} else {
|
|
return parent::columnToProperty($column);
|
|
}
|
|
}
|
|
|
|
public function propertyToColumn($property) {
|
|
if ($property === 'phoneNumber') {
|
|
return 'phonenumber';
|
|
} else {
|
|
return parent::propertyToColumn($property);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
Slugs
|
|
^^^^^
|
|
|
|
Slugs are used to identify resources in the URL by a string rather than integer id. Since the URL allows only certain values, the entity base class provides a slugify method for it:
|
|
|
|
.. code-block:: php
|
|
|
|
<?php
|
|
$author = new Author();
|
|
$author->setName('Some*thing');
|
|
$author->slugify('name'); // Some-thing
|
|
|
|
|