diff --git a/manual/docs/en.txt b/manual/docs/en.txt index 05f0324f8..94ec7e1a3 100644 --- a/manual/docs/en.txt +++ b/manual/docs/en.txt @@ -5,6 +5,7 @@ + Basic schema mapping + Relations + Working with objects ++ Component overview + Hierarchical data + Configuration + DQL (Doctrine Query Language) diff --git a/manual/docs/en/component-overview.txt b/manual/docs/en/component-overview.txt new file mode 100644 index 000000000..e963e8018 --- /dev/null +++ b/manual/docs/en/component-overview.txt @@ -0,0 +1,715 @@ +++ Record +{{Doctrine_Record}} is one of the most essential components of Doctrine ORM. The class is a wrapper for database row but along with that it speficies what relations it has +on other components and what columns it has. It may access the related components, hence its refered as an ActiveRecord. + +The classes that inherit {{Doctrine_Record}} are called components. There should be atleast one component for each database table. + +There are couple of ways for creating new records. Propably the easiest is using native php new -operator. The other ways are calling {{Doctrine_Table::create()}} or {{Doctrine_Connection::create()}}. The last two exists only for backward compatibility. The recommended way of creating new objects is the new operator. + + +$user = $conn->create("User"); + +// alternative way: + +$table = $conn->getTable("User"); + +$user = $table->create(); + +// the simpliest way: + +$user = new User(); + +// records support array access +$user['name'] = 'John Locke'; + +// save user into database +$user->save(); + + +Every record has an object identifier, which is an internal unique identifier. You can get the object identifier with the oid() method. Basically two objects are considered the same if they share the same object identifier. + + ++++ Properties + +Each assigned column property of {{Doctrine_Record}} represents a database table column. As you've learned in the previous chapters the column definitions can be achieved with the hasColumn() method. Now accessing the columns is easy. You can use any of the means described above. The recommended way is using the ArrayAccess as it makes it easy to switch between record and array fetching when needed. + + +$user = $table->find(3); + +// access property through overloading + +$name = $user->name; + +// access property with get() + +$name = $user->get("name"); + +// access property with ArrayAccess interface + +$name = $user['name']; + + +Iterating trhough the properties of a record can be done in similar way as iterating through an array - by using the foreach construct. This is possible since {{Doctrine_Record}} implements a magic IteratorAggregate interface. + + +foreach ($user as $field => $value) { + +} + + +As with arrays you can use the isset() for checking if given property exists and unset() for setting given property to null. + + +// checking if property called 'name' exists +if (isset($user['name'])) { + +} + +// unsetting name property +unset($user['name']); + + +When you have set values for record properties you can get the names of modified properties with modifiedFields() method. This method returns an array of modified field names. + + +$user['name'] = 'Jack Daniels'; + +$user['age'] = 100; + +print_r($user->modifiedFields()); // array('name', 'age'); + +$user->isModified(); // true + + +Sometimes you may want to retrieve the column count of given record. In order to do this you can simply pass the record as an argument for the count() function. This is possible since {{Doctrine_Record}} implements a magic Countable interface. The other way would be calling the count() method. + + +// get the number of columns + +$colCount = $record->count(); + +$colCount = count($record); + + + +{{Doctrine_Record}} offers a special method for accessing the identifier of given record. This method is called identifier() and it returns an array with identifier field names as keys and values as the associated property values. + +$user['name'] = 'Jack Daniels'; + +$user->save(); + +$user->identifier(); // array('id' => 1) + + +A common case is that you have an array of values which you need to assign to a given record. It may feel awkward and clumsy to set these values separately. No need to worry though, {{Doctrine_Record}} offers a way for merging given array to property values. + +The merge() method iterates through the properties of given record and assigns the values of given array to the associated properties. + + +$values = array('name' => 'someone', + 'age' => 11, + 'unknownproperty' => '...'); + +// notice that here the unknownproperty won't get assigned +// as the User class doesn't have a column with that name +$user->merge($values); + +print $user->name; // someone +print $user->age; // 11 + +print $user->unknownproperty; // throws exception + + + ++++ Retrieving existing records + +Doctrine provides many ways for record retrieval. The fastest ways for retrieving existing records are the finder methods provided by {{Doctrine_Table}}. If you need to use more complex queries take a look at the DQL API. + + +$table = $conn->getTable("User"); + +// find by primary key + +$user = $table->find(2); +if($user !== false) + print $user->name; + +// get all users +foreach($table->findAll() as $user) { + print $user->name; +} + +// finding by dql +foreach($table->findByDql("name LIKE '%John%'") as $user) { + print $user->created; +} + +// finding objects with DQL + +$users = $conn->query("FROM User u WHERE u.name LIKE '%John%'"); + + + + + ++++ Updating records + +Updating objects is very easy, you just call the {{Doctrine_Record::save()}} method. The other way is to call {{Doctrine_Connection::flush()}} which saves all objects. It should be noted though that flushing is a much heavier operation than just calling save method. + + +$table = $conn->getTable('User'); + +$user = $table->find(2); + +if($user !== false) { + $user->name = 'Jack Daniels'; + + $user->save(); +} + + +Sometimes you may want to do a direct update. In direct update the objects aren't loaded from database, rather the state of the database is directly updated. In the following example we use DQL UPDATE statement to update all users. + + + +// make all usernames lowercased +Doctrine_Query::create()->update('User u') + ->set('u.name', 'LOWER(u.name)') + ->execute(); + + ++++ Deleting records + +Deleting records in Doctrine is handled by {{Doctrine_Record::delete()}}, {{Doctrine_Collection::delete()}} and {{Doctrine_Connection::delete()}} methods. + + +$table = $conn->getTable("User"); + +$user = $table->find(2); + +// deletes user and all related composite objects +if($user !== false) + $user->delete(); + + +$users = $table->findAll(); + + +// delete all users and their related composite objects +$users->delete(); + + ++++ Using expression values + +There might be situations where you need to use SQL expressions as values of columns. This can be achieved by using Doctrine_Expression which converts portable DQL expressions to your native SQL expressions. + +Lets say we have a class called event with columns timepoint(datetime) and name(string). Saving the record with the current timepoint can be achieved as follows: + +$event = new Event(); +$event->name = 'Rock festival'; +$event->timepoint = new Doctrine_Expression('NOW()'); + +$event->save(); + + +The last line would execute sql (in sqlite): + +INSERT INTO event (name, timepoint) VALUES (?, 'NOW()') + + ++++ Getting record state + +{{Every Doctrine_Record}} has a state. First of all records can be transient or persistent. Every record that is retrieved from database is persistent and every newly created record is considered transient. If a {{Doctrine_Record}} is retrieved from database but the only loaded property is its primary key, then this record has a state called proxy. + +Every transient and persistent {{Doctrine_Record}} is either clean or dirty. {{Doctrine_Record}} is clean when none of its properties are changed and dirty when atleast one of its properties has changed. + +A record can also have a state called locked. In order to avoid infinite recursion in some rare circular reference cases Doctrine uses this state internally to indicate that a record is currently under a manipulation operation. + + +$state = $record->state(); + +switch($state): + case Doctrine_Record::STATE_PROXY: + // record is in proxy state, + // meaning its persistent but not all of its properties are + // loaded from the database + break; + case Doctrine_Record::STATE_TCLEAN: + // record is transient clean, + // meaning its transient and + // none of its properties are changed + break; + case Doctrine_Record::STATE_TDIRTY: + // record is transient dirty, + // meaning its transient and + // some of its properties are changed + break; + case Doctrine_Record::STATE_DIRTY: + // record is dirty, + // meaning its persistent and + // some of its properties are changed + break; + case Doctrine_Record::STATE_CLEAN: + // record is clean, + // meaning its persistent and + // none of its properties are changed + break; + case Doctrine_Record::STATE_LOCKED: + // record is locked + break; +endswitch; + + + ++++ Getting object copy + +Sometimes you may want to get a copy of your object (a new object with all properties copied). Doctrine provides a simple method for this: {{Doctrine_Record::copy()}}. + + +// get a simple copy +$copy = $user->copy(); + + +Notice that copying the record with copy() returns a new record (state TDIRTY) with the values of the old record, but it does *not* copy the relations of that record. If you want to copy the relations too, you need to use copyDeep(). + + +// get a copy of user with all the relations +$copy = $user->copyDeep(); + + ++++ Saving a blank record + +By default Doctrine doesn't execute when save() is being called on an unmodified record. There might be situations where you want to force-insert the record even if it has not been modified. This can be achieved by assigning the state of the record to Doctrine_Record::STATE_TDIRTY. + + +$user = new User(); +$user->state('TDIRTY'); +$user->save(); + +$user->id; // 1 + + ++++ Mapping custom values + +There might be situations where you want to map custom values to records. For example values that depend on some outer sources and you only want these values to be availible at runtime not persisting those values into database. This can be achieved as follows: + + +$user->mapValue('isRegistered', true); + +$user->isRegistered; // true + + + ++++ Serializing + +Sometimes you may want to serialize your record objects (possibly for caching purposes). Records can be serialized, but remember: Doctrine cleans all relations, before doing this. So remember to persist your objects into database before serializing them. + + +$string = serialize($user); + +$user = unserialize($string); + + + ++++ Checking existence + +Very commonly you'll need to know if given record exists in the database. You can use the exists() method for checking if given record has a database row equivalent. + + +$record = new User(); + +$record->exists(); // false + +$record->name = 'someone'; +$record->save(); + +$record->exists(); // true + + ++++ Function callbacks for columns + +{{Doctrine_Record}} offers a way for attaching callback calls for column values. For example if you want to trim certain column, you can simply type: + + +$record->call('trim', 'column1'); + + +++ Collection + +{{Doctrine_Collection}} is a collection of records (see {{Doctrine_Record}}). As with records the collections can be deleted and saved using {{Doctrine_Collection::delete()}} and {{Doctrine_Collection::save()}} accordingly. + +When fetching data from database with either DQL API (see {{Doctrine_Query}}) or rawSql API (see {{Doctrine_RawSql}}) the methods return an instance of {{Doctrine_Collection}} by default. + +The following example shows how to initialize a new collection: + + +$conn = Doctrine_Manager::getInstance() + ->openConnection(new PDO("dsn", "username", "pw")); + +// initalizing a new collection +$users = new Doctrine_Collection($conn->getTable('User')); + +// alternative (propably easier) +$users = new Doctrine_Collection('User'); + +// adding some data +$users[0]->name = 'Arnold'; + +$users[1]->name = 'Somebody'; + +// finally save it! +$users->save(); + + + ++++ Accessing elements + +You can access the elements of {{Doctrine_Collection}} with {{set()}} and {{get()}} methods or with {{ArrayAccess}} interface. + + +$table = $conn->getTable("User"); + +$users = $table->findAll(); + +// accessing elements with ArrayAccess interface + +$users[0]->name = "Jack Daniels"; + +$users[1]->name = "John Locke"; + +// accessing elements with get() + +print $users->get(1)->name; + + + ++++ Adding new elements + +When accessing single elements of the collection and those elements (records) don't exist Doctrine auto-adds them. + +In the following example we fetch all users from database (there are 5) and then add couple of users in the collection. + +As with PHP arrays the indexes start from zero. + + +$users = $table->findAll(); + +print count($users); // 5 + +$users[5]->name = "new user 1"; +$users[6]->name = "new user 2"; + + + ++++ Getting collection count + +The {{Doctrine_Collection}} method {{count()}} returns the number of elements currently in the collection. + + +$users = $table->findAll(); + +print $users->count(); + + +Since {{Doctrine_Collection}} implements Countable interface a valid alternative for the previous example is to simply pass the collection as an argument for the count() function. + + +print count($users); // Doctrine_Collection implements Countable interface + + + ++++ Saving the collection + +Similar to {{Doctrine_Record}} the collection can be saved by calling the {{save()}} method. When save() gets called Doctrine issues save() operations an all records and wraps the whole procedure in a transaction. + + +$users = $table->findAll(); + +$users[0]->name = 'Jack Daniels'; + +$users[1]->name = 'John Locke'; + +$users->save(); + + + ++++ Deleting collection + +Doctrine Collections can be deleted in very same way is Doctrine Records you just call {{delete()}} method. As for all collections Doctrine knows how to perform single-shot-delete meaning it only performs one database query for the each collection. + +For example if we have collection of users. When deleting the collection +of users doctrine only performs one query for this whole transaction. The query would look something like: + + +DELETE FROM user WHERE id IN (1,2,3, ... ,N) + + ++++ Key mapping + +Sometimes you may not want to use normal indexing for collection elements. For example in some cases mapping primary keys as collection keys might be useful. The following example demonstrates how this can be achieved. + + +// mapping id column + +$user = new User(); + +$user->setAttribute(Doctrine::ATTR_COLL_KEY, 'id'); + +// now user collections will use the values of +// id column as element indexes + +$users = $user->getTable()->findAll(); + +foreach($users as $id => $user) { + print $id . $user->name; +} + +// mapping name column + +$user = new User(); + +$user->setAttribute(Doctrine::ATTR_COLL_KEY, 'name'); + +// now user collections will use the values of +// name column as element indexes + +$users = $user->getTable()->findAll(); + +foreach($users as $name => $user) { + print $name . $user->type; +} + + + ++++ Loading related records + +Doctrine provides means for efficiently retrieving all related records for all record elements. That means when you have for example a collection of users you can load all phonenumbers for all users by simple calling the {{loadRelated()}} method. + +However, in most cases you don't need to load related elements explicitly, rather what you should do is try to load everything at once by using the DQL API and JOINS. + +The following example uses three queries for retrieving users, their phonenumbers and the groups they belong to. + + +$users = $conn->query('FROM User'); + +// now lets load phonenumbers for all users + +$users->loadRelated('Phonenumber'); + +foreach($users as $user) { + print $user->Phonenumber[0]->phonenumber; + // no additional db queries needed here +} + +// the loadRelated works an any relation, even associations: + +$users->loadRelated('Group'); + +foreach($users as $user) { + print $user->Group[0]->name; +} + + +The example above shows how to do this more efficiently by using the DQL API. + + +// load everything here +$users = $conn->query('FROM User u LEFT JOIN u.Phonenumber p LEFT JOIN u.Group g'); + +foreach($users as $user) { + // no additional db queries needed here + print $user->Phonenumber->phonenumber; + + print $user->Group->name; +} + + +++ Connection + +Doctrine_Connection is a wrapper for database connection. It handles several things: + +* Handles database portability things missing from PDO (eg. {{LIMIT}} / {{OFFSET}} emulation) +* Keeps track of {{Doctrine_Table}} objects +* Keeps track of records +* Keeps track of records that need to be updated / inserted / deleted +* Handles transactions and transaction nesting +* Handles the actual querying of the database in the case of {{INSERT}} / {{UPDATE}} / {{DELETE}} operations +* Can query the database using the DQL API (see {{Doctrine_Query}}) +* Optionally validates transactions using {{Doctrine_Validator}} and gives full information of possible errors. + + ++++ Available drivers + +Doctrine has drivers for every PDO-supported database. The supported databases are: + +* FreeTDS / Microsoft SQL Server / Sybase +* Firebird/Interbase 6 +* Informix +* Mysql +* Oracle +* Odbc +* PostgreSQL +* Sqlite + + ++++ Getting a table object + +In order to get table object for specified record just call {{Doctrine_Record::getTable()}} or {{Doctrine_Connection::getTable()}}. + + +$manager = Doctrine_Manager::getInstance(); + +// open new connection + +$conn = $manager->openConnection(new PDO('dsn','username','password')); + +// getting a table object + +$table = $conn->getTable('User'); + + + ++++ Flushing the connection + +Creating new record (database row) is very easy. You can either use the {{Doctrine_Connection::create()}} or {{Doctrine_Table::create()}} method to do this or just simply use the new operator. + + +$user = new User(); +$user->name = 'Jack'; + +$group = $conn->create('Group'); +$group->name = 'Drinking Club'; + +// saves all the changed objects into database + +$conn->flush(); + + + ++++ Querying the database + +{{Doctrine_Connection::query()}} is a simple method for efficient object retrieval. It takes one parameter (DQL query) and optionally prepared statement params. + + + +// select all users + +$users = $conn->query('FROM User'); + +// select all users where user email is jackdaniels@drinkmore.info + +$users = $conn->query("FROM User WHERE User.Email.address = 'jackdaniels@drinkmore.info'"); + +// using prepared statements + +$users = $conn->query('FROM User WHERE User.name = ?', array('Jack')); + + + + +++ Table +{{Doctrine_Table}} holds the schema information specified by the given component (record). For example if you have a User class that extends Doctrine_Record, each schema definition call gets delegated to a unique table object that holds the information for later use. + +Each {{Doctrine_Table}} is registered by {{Doctrine_Connection}}, which means you can retrieve the tables from the connection by calling the getTable() method with the appropriate component name. + +For example, lets say we want to retrieve the table object for the User class. We can do this by simply giving the 'User' as the first argument for the getTable() method. + + +// get the current connection +$conn = Doctrine_Manager::connection(); + +$table = $conn->getTable('User'); + + ++++ Getting column information + +You can retrieve the column definitions set in {{Doctrine_Record}} by using the appropriate {{Doctrine_Table}} methods. If you need all information of all columns you can simply use: + + +// getting all information of all columns +$columns = $table->getColumns(); + + +Sometimes this can be an overkill. The following example shows how to retrieve the column names as an array: + + +// getting column names +$names = $table->getColumnNames(); + + ++++ Getting relation information + + ++++ Finder methods + +{{Doctrine_Table}} provides basic finder methods. These finder methods are very fast and should be used if you only need to fetch data from one database table. If you need queries that use several components (database tables) use {{Doctrine_Connection::query()}}. + + +$table = $conn->getTable("User"); + +// find by primary key + +$user = $table->find(2); + +if($user !== false) + print $user->name; + + +// get all users +foreach($table->findAll() as $user) { + print $user->name; +} + +// finding by dql +foreach($table->findByDql("name LIKE '%John%'") as $user) { + print $user->created; +} + + + +++++ Custom table classes + +Adding custom table classes is very easy. Only thing you need to do is name the classes as {{[componentName]Table}} and make them inherit {{Doctrine_Table}}. + + + +// valid table object + +class UserTable extends Doctrine_Table +{ +} + +// not valid [doesn't extend Doctrine_Table] +class GroupTable { } + + + ++++ Custom finders + +You can add custom finder methods to your custom table object. These finder methods may use fast {{Doctrine_Table}} finder methods or DQL API ({{Doctrine_Connection::query()}}). + + +class UserTable extends Doctrine_Table { + /** + * you can add your own finder methods here + */ + public function findByName($name) { + return $this->getConnection()->query("FROM User WHERE name LIKE '%$name%'"); + } +} +class User extends Doctrine_Record { } + +$conn = Doctrine_Manager::getInstance() + ->openConnection(new PDO("dsn","username","password")); + +// doctrine will now check if a class called UserTable exists +// and if it inherits Doctrine_Table + +$table = $conn->getTable("User"); + +print get_class($table); // UserTable + +$users = $table->findByName("Jack"); + + + diff --git a/manual/docs/en/working-with-objects.txt b/manual/docs/en/working-with-objects.txt index 84bdd52ad..9cd8fc5a2 100644 --- a/manual/docs/en/working-with-objects.txt +++ b/manual/docs/en/working-with-objects.txt @@ -56,8 +56,6 @@ $user->save(); This is due to a fact that $user->GroupUser loads all group links for given user. This can time-consuming task if user belongs to many groups. Even if the user belongs to few groups this will still execute an unnecessary SELECT statement. - -++ Component overview ++ Fetching objects Normally when you fetch data from database the following phases are executed: @@ -130,7 +128,7 @@ This structure also applies to the hydration of objects(records) which is the de You should always use array hydration when you only need to data for access-only purposes, whereas you should use the record hydration when you need to change the fetched data. -The constant O(n) performance of the hydration algorithm is ensured by a smart identifier caching solution. +The constant O(n) performance of the hydration algorithm is ensured by a smart identifier caching solution. +++ Field lazy-loading diff --git a/manual/docs/en/working-with-objects/component-overview.txt b/manual/docs/en/working-with-objects/component-overview.txt deleted file mode 100644 index c6fcbd172..000000000 --- a/manual/docs/en/working-with-objects/component-overview.txt +++ /dev/null @@ -1,5 +0,0 @@ -+++ Manager -+++ Connection -+++ Record -+++ Collection -+++ Table diff --git a/manual/docs/en/working-with-objects/component-overview/collection.txt b/manual/docs/en/working-with-objects/component-overview/collection.txt index ebaf38b12..d3f5a12fa 100644 --- a/manual/docs/en/working-with-objects/component-overview/collection.txt +++ b/manual/docs/en/working-with-objects/component-overview/collection.txt @@ -1,184 +1 @@ -++++ Introduction -{{Doctrine_Collection}} is a collection of records (see {{Doctrine_Record}}). As with records the collections can be deleted and saved using {{Doctrine_Collection::delete()}} and {{Doctrine_Collection::save()}} accordingly. - -When fetching data from database with either DQL API (see {{Doctrine_Query}}) or rawSql API (see {{Doctrine_RawSql}}) the methods return an instance of {{Doctrine_Collection}} by default. - -The following example shows how to initialize a new collection: - - -$conn = Doctrine_Manager::getInstance() - ->openConnection(new PDO("dsn", "username", "pw")); - -// initalizing a new collection -$users = new Doctrine_Collection($conn->getTable('User')); - -// alternative (propably easier) -$users = new Doctrine_Collection('User'); - -// adding some data -$users[0]->name = 'Arnold'; - -$users[1]->name = 'Somebody'; - -// finally save it! -$users->save(); - - - -++++ Accessing elements - -You can access the elements of {{Doctrine_Collection}} with {{set()}} and {{get()}} methods or with {{ArrayAccess}} interface. - - -$table = $conn->getTable("User"); - -$users = $table->findAll(); - -// accessing elements with ArrayAccess interface - -$users[0]->name = "Jack Daniels"; - -$users[1]->name = "John Locke"; - -// accessing elements with get() - -print $users->get(1)->name; - - - -++++ Adding new elements - -When accessing single elements of the collection and those elements (records) don't exist Doctrine auto-adds them. - -In the following example we fetch all users from database (there are 5) and then add couple of users in the collection. - -As with PHP arrays the indexes start from zero. - - -$users = $table->findAll(); - -print count($users); // 5 - -$users[5]->name = "new user 1"; -$users[6]->name = "new user 2"; - - - -++++ Getting collection count - -The {{Doctrine_Collection}} method {{count()}} returns the number of elements currently in the collection. - - -$users = $table->findAll(); - -$users->count(); - -// or - -count($users); // Doctrine_Collection implements Countable interface - - - -++++ Saving the collection - -As with records the collection can be saved by calling the save method. - - -$users = $table->findAll(); - -$users[0]->name = "Jack Daniels"; - -$users[1]->name = "John Locke"; - -$users->save(); - - - -++++ Deleting collection - -Doctrine Collections can be deleted in very same way is Doctrine Records you just call {{delete()}} method. As for all collections Doctrine knows how to perform single-shot-delete meaning it only performs one database query for the each collection. - -For example if we have collection of users which own [0-*] phonenumbers. When deleting the collection -of users doctrine only performs two queries for this whole transaction. The queries would look something like: - - -DELETE FROM user WHERE id IN (1,2,3, ... ,N) -DELETE FROM phonenumber WHERE id IN (1,2,3, ... ,M) - - -It should also be noted that Doctrine is smart enough to perform single-shot-delete per table when transactions are used. So if you are deleting a lot of records and want to optimize the operation just wrap the delete calls in {{Doctrine_Connection}} transaction. - - -// delete all users with name 'John' - -$users = $table->findByDql("name LIKE '%John%'"); - -$users->delete(); - - - -++++ Key mapping - -Sometimes you may not want to use normal indexing for collection elements. For example in some cases mapping primary keys as collection keys might be useful. The following example demonstrates how this can be achieved. - - -// mapping id column - -$user = new User(); - -$user->setAttribute(Doctrine::ATTR_COLL_KEY, 'id'); - -// now user collections will use the values of -// id column as element indexes - -$users = $user->getTable()->findAll(); - -foreach($users as $id => $user) { - print $id . $user->name; -} - -// mapping name column - -$user = new User(); - -$user->setAttribute(Doctrine::ATTR_COLL_KEY, 'name'); - -// now user collections will use the values of -// name column as element indexes - -$users = $user->getTable()->findAll(); - -foreach($users as $name => $user) { - print $name . $user->type; -} - - - -++++ Loading related records - -Doctrine provides means for efficiently retrieving all related records for all record elements. That means when you have for example a collection of users you can load all phonenumbers for all users by simple calling the {{loadRelated()}} method. - - -$users = $conn->query("FROM User"); - -// now lets load phonenumbers for all users - -$users->loadRelated("Phonenumber"); - -foreach($users as $user) { - print $user->Phonenumber->phonenumber; - // no additional db queries needed here -} - -// the loadRelated works an any relation, even associations: - -$users->loadRelated("Group"); - -foreach($users as $user) { - print $user->Group->name; -} - - - -++++ Collection expanding diff --git a/manual/docs/en/working-with-objects/component-overview/connection.txt b/manual/docs/en/working-with-objects/component-overview/connection.txt index 437f6f19b..99a809136 100644 --- a/manual/docs/en/working-with-objects/component-overview/connection.txt +++ b/manual/docs/en/working-with-objects/component-overview/connection.txt @@ -1,103 +1,2 @@ -++++ Introduction - -Doctrine_Connection is a wrapper for database connection. It handles several things: - -* Handles database portability things missing from PDO (eg. {{LIMIT}} / {{OFFSET}} emulation) -* Keeps track of {{Doctrine_Table}} objects -* Keeps track of records -* Keeps track of records that need to be updated / inserted / deleted -* Handles transactions and transaction nesting -* Handles the actual querying of the database in the case of {{INSERT}} / {{UPDATE}} / {{DELETE}} operations -* Can query the database using the DQL API (see {{Doctrine_Query}}) -* Optionally validates transactions using {{Doctrine_Validator}} and gives full information of possible errors. - -++++ Available drivers - -Doctrine has drivers for every PDO-supported database. The supported databases are: - -* FreeTDS / Microsoft SQL Server / Sybase -* Firebird/Interbase 6 -* Informix -* Mysql -* Oracle -* Odbc -* PostgreSQL -* Sqlite - - -++++ Getting a table object - -In order to get table object for specified record just call {{Doctrine_Record::getTable()}} or {{Doctrine_Connection::getTable()}}. - - -$manager = Doctrine_Manager::getInstance(); - -// open new connection - -$conn = $manager->openConnection(new PDO('dsn','username','password')); - -// getting a table object - -$table = $conn->getTable('User'); - - - -++++ Flushing the connection - -Creating new record (database row) is very easy. You can either use the {{Doctrine_Connection::create()}} or {{Doctrine_Table::create()}} method to do this or just simply use the new operator. - - -$user = new User(); -$user->name = 'Jack'; - -$group = $conn->create('Group'); -$group->name = 'Drinking Club'; - -// saves all the changed objects into database - -$conn->flush(); - - - -++++ Querying the database - -{{Doctrine_Connection::query()}} is a simple method for efficient object retrieval. It takes one parameter (DQL query) and optionally prepared statement params. - - - -// select all users - -$users = $conn->query('FROM User'); - -// select all users where user email is jackdaniels@drinkmore.info - -$users = $conn->query("FROM User WHERE User.Email.address = 'jackdaniels@drinkmore.info'"); - -// using prepared statements - -$users = $conn->query('FROM User WHERE User.name = ?', array('Jack')); - - - -++++ Getting connection state - -Connection state gives you information about how active connection currently is. You can get the current state by calling {{Doctrine_Connection::getState()}}. - - -switch($conn->getState()): - case Doctrine_Connection::STATE_ACTIVE: - // connection open and zero open transactions - break; - case Doctrine_Connection::STATE_ACTIVE: - // one open transaction - break; - case Doctrine_Connection::STATE_BUSY: - // multiple open transactions - break; - case Doctrine_Connection::STATE_CLOSED: - // connection closed - break; -endswitch; - diff --git a/manual/docs/en/working-with-objects/component-overview/record.txt b/manual/docs/en/working-with-objects/component-overview/record.txt index 3780b7c8d..d3f5a12fa 100644 --- a/manual/docs/en/working-with-objects/component-overview/record.txt +++ b/manual/docs/en/working-with-objects/component-overview/record.txt @@ -1,243 +1 @@ -++++ Introduction -{{Doctrine_Record}} is a wrapper for database row but along with that it speficies what relations it has -on other components and what columns it has. It may access the related components, hence its refered as an ActiveRecord. - -The classes that inherit {{Doctrine_Record}} are called components. There should be atleast one component for each database table. - - -++++ Creating new records - -There are couple of ways for creating new records. Propably the easiest is using native php new -operator. The other ways are calling {{Doctrine_Table::create()}} or {{Doctrine_Connection::create()}}. The last two exists only for backward compatibility. The recommended way of creating new objects is the new operator. - - -$user = $conn->create("User"); - -// alternative way: - -$table = $conn->getTable("User"); - -$user = $table->create(); - -// the simpliest way: - -$user = new User(); - - -// records support array access -$user["name"] = "John Locke"; - -// save user into database -$user->save(); - - - -++++ Retrieving existing records - -Doctrine provides many ways for record retrieval. The fastest ways for retrieving existing records are the finder methods provided by {{Doctrine_Table}}. If you need to use more complex queries take a look at DQL API and {{Doctrine_Connection::query}} method. - - -$table = $conn->getTable("User"); - -// find by primary key - -$user = $table->find(2); -if($user !== false) - print $user->name; - -// get all users -foreach($table->findAll() as $user) { - print $user->name; -} - -// finding by dql -foreach($table->findByDql("name LIKE '%John%'") as $user) { - print $user->created; -} - -// finding objects with DQL - -$users = $conn->query("FROM User WHERE User.name LIKE '%John%'"); - - - -++++ Accessing properties - -You can retrieve existing objects (database rows) with {{Doctrine_Table}} or {{Doctrine_Connection}}. {{Doctrine_Table}} provides simple methods like {{findBySql}}, {{findAll}} and find for finding objects whereas {{Doctrine_Connection}} provides complete OQL API for retrieving objects (see chapter [doc dql-doctrine-query-language). - - -$user = $table->find(3); - -// access property through overloading - -$name = $user->name; - -// access property with get() - -$name = $user->get("name"); - -// access property with ArrayAccess interface - -$name = $user['name']; - -// iterating through properties - -foreach($user as $key => $value) { - -} - - - -++++ Updating records - -Updating objects is very easy, you just call the {{Doctrine_Record::save()}} method. The other way (perhaps even easier) is to call {{Doctrine_Connection::flush()}} which saves all objects. It should be noted though that flushing is a much heavier operation than just calling save method. - - -$table = $conn->getTable('User'); - - -$user = $table->find(2); - -if($user !== false) { - $user->name = 'Jack Daniels'; - - $user->save(); -} - - - -++++ Deleting records - -Deleting records in Doctrine is handled by {{Doctrine_Record::delete()}}, {{Doctrine_Collection::delete()}} and {{Doctrine_Connection::delete()}} methods. - - -$table = $conn->getTable("User"); - -$user = $table->find(2); - -// deletes user and all related composite objects -if($user !== false) - $user->delete(); - - -$users = $table->findAll(); - - -// delete all users and their related composite objects -$users->delete(); - - -++++ Using expression values - -There might be situations where you need to use SQL expressions as values of columns. This can be achieved by using Doctrine_Expression which converts portable DQL expressions to your native SQL expressions. - -Lets say we have a class called event with columns timepoint(datetime) and name(string). Saving the record with the current timepoint can be achieved as follows: - -$event = new Event(); -$event->name = 'Rock festival'; -$event->timepoint = new Doctrine_Expression('NOW()'); - -$event->save(); - - -The last line would execute sql (in sqlite): - -INSERT INTO event (name, timepoint) VALUES (?, 'NOW()') - - -++++ Getting record state - -{{Every Doctrine_Record}} has a state. First of all record can be transient or persistent. Every record that is retrieved from database is persistent and every newly created record is transient. If a {{Doctrine_Record}} is retrieved from database but the only loaded property is its primary key, then this record has a state called proxy. - -Every transient and persistent {{Doctrine_Record}} is either clean or dirty. {{Doctrine_Record}} is clean when none of its properties are changed and dirty when atleast one of its properties has changed. - - -$state = $record->state(); - -switch($state): - case Doctrine_Record::STATE_PROXY: - // record is in proxy state, - // meaning its persistent but not all of its properties are - // loaded from the database - break; - case Doctrine_Record::STATE_TCLEAN: - // record is transient clean, - // meaning its transient and - // none of its properties are changed - break; - case Doctrine_Record::STATE_TDIRTY: - // record is transient dirty, - // meaning its transient and - // some of its properties are changed - break; - case Doctrine_Record::STATE_DIRTY: - // record is dirty, - // meaning its persistent and - // some of its properties are changed - break; - case Doctrine_Record::STATE_CLEAN: - // record is clean, - // meaning its persistent and - // none of its properties are changed - break; -endswitch; - - - -++++ Getting object copy - -Sometimes you may want to get a copy of your object (a new object with all properties copied). Doctrine provides a simple method for this: {{Doctrine_Record::copy()}}. - - -$copy = $user->copy(); - - -++++ Saving a blank record - -By default Doctrine doesn't execute when save() is being called on an unmodified record. There might be situations where you want to force-insert the record even if it has not been modified. This can be achieved by assigning the state of the record to Doctrine_Record::STATE_TDIRTY. - - -$user = new User(); -$user->state('TDIRTY'); -$user->save(); - -$user->id; // 1 - - -++++ Mapping custom values - -There might be situations where you want to map custom values to records. For example values that depend on some outer sources and you only want these values to be availible at runtime not persisting those values into database. This can be achieved as follows: - - -$user->mapValue('isRegistered', true); - -$user->isRegistered; // true - - - -++++ Serializing - -Sometimes you may want to serialize your record objects (possibly for caching purposes). Records can be serialized, but remember: Doctrine cleans all relations, before doing this. So remember to persist your objects into database before serializing them. - - -$string = serialize($user); - -$user = unserialize($string); - - - -++++ Checking Existence - - -$record = new User(); - -$record->exists(); // false - -$record->name = 'someone'; -$record->save(); - -$record->exists(); // true - - - -++++ Callbacks