diff --git a/manual/codes/Advanced components - Eventlisteners - Creating new listener.php b/manual/codes/Advanced components - Eventlisteners - Creating new listener.php
new file mode 100644
index 000000000..4ae75f0d7
--- /dev/null
+++ b/manual/codes/Advanced components - Eventlisteners - Creating new listener.php
@@ -0,0 +1,34 @@
+getTable()->getComponentName()." just got loaded!";
+ }
+ public function onSave(Doctrine_Record $record) {
+ print "saved data access object!";
+ }
+}
+class MyListener2 extends Doctrine_EventListener {
+ public function onPreUpdate() {
+ try {
+ $record->set("updated",time());
+ } catch(InvalidKeyException $e) {
+ }
+ }
+}
+
+
+// setting global listener
+$manager = Doctrine_Manager::getInstance();
+
+$manager->setAttribute(Doctrine::ATTR_LISTENER,new MyListener());
+
+// setting session level listener
+$session = $manager->openSession($dbh);
+
+$session->setAttribute(Doctrine::ATTR_LISTENER,new MyListener2());
+
+// setting factory level listener
+$table = $session->getTable("User");
+
+$table->setAttribute(Doctrine::ATTR_LISTENER,new MyListener());
+?>
diff --git a/manual/codes/Advanced components - Eventlisteners - List of events.php b/manual/codes/Advanced components - Eventlisteners - List of events.php
new file mode 100644
index 000000000..daf382388
--- /dev/null
+++ b/manual/codes/Advanced components - Eventlisteners - List of events.php
@@ -0,0 +1,45 @@
+
diff --git a/manual/codes/Advanced components - Eventlisteners - Listening events.php b/manual/codes/Advanced components - Eventlisteners - Listening events.php
new file mode 100644
index 000000000..f3f09f616
--- /dev/null
+++ b/manual/codes/Advanced components - Eventlisteners - Listening events.php
@@ -0,0 +1,13 @@
+getTable("User");
+
+$table->setEventListener(new MyListener2());
+
+// retrieve user whose primary key is 2
+$user = $table->find(2);
+
+$user->name = "John Locke";
+
+// update event will be listened and current time will be assigned to the field 'updated'
+$user->save();
+?>
diff --git a/manual/codes/Advanced components - Hook - Parameter hooking.php b/manual/codes/Advanced components - Hook - Parameter hooking.php
new file mode 100644
index 000000000..44eb80df5
--- /dev/null
+++ b/manual/codes/Advanced components - Hook - Parameter hooking.php
@@ -0,0 +1,3 @@
+
diff --git a/manual/codes/Advanced components - Validators - Analyzing the ErrorStack.php b/manual/codes/Advanced components - Validators - Analyzing the ErrorStack.php
new file mode 100644
index 000000000..8e6097eb8
--- /dev/null
+++ b/manual/codes/Advanced components - Validators - Analyzing the ErrorStack.php
@@ -0,0 +1,27 @@
+name = "this is an example of too long name";
+ $user->Email->address = "drink@@notvalid..";
+ $user->save();
+} catch(Doctrine_Validator_Exception $e) {
+ $stack = $e->getErrorStack();
+ foreach($stack as $component => $err) {
+ foreach($err as $field => $type) {
+ switch($type):
+ case Doctrine_Validator::ERR_TYPE:
+ print $field." is not right type";
+ break;
+ case Doctrine_Validator::ERR_UNIQUE:
+ print $field." is not unique";
+ break;
+ case Doctrine_Validator::ERR_VALID:
+ print $field." is not valid";
+ break;
+ case Doctrine_Validator::ERR_LENGTH:
+ print $field." is too long";
+ break;
+ endswitch;
+ }
+ }
+}
+?>
diff --git a/manual/codes/Advanced components - Validators - Intruduction.php b/manual/codes/Advanced components - Validators - Intruduction.php
new file mode 100644
index 000000000..be47e66bd
--- /dev/null
+++ b/manual/codes/Advanced components - Validators - Intruduction.php
@@ -0,0 +1,5 @@
+setAttribute(Doctrine::ATTR_VLD, true);
+?>
diff --git a/manual/codes/Advanced components - Validators - Validating transactions.php b/manual/codes/Advanced components - Validators - Validating transactions.php
new file mode 100644
index 000000000..419f20310
--- /dev/null
+++ b/manual/codes/Advanced components - Validators - Validating transactions.php
@@ -0,0 +1,42 @@
+ownsOne("Email","User.email_id");
+ }
+ public function setTableDefinition() {
+ // no special validators used only types
+ // and lengths will be validated
+ $this->hasColumn("name","string",15);
+ $this->hasColumn("email_id","integer");
+ $this->hasColumn("created","integer",11);
+ }
+}
+class Email extends Doctrine_Record {
+ public function setTableDefinition() {
+ // specialized validators 'email' and 'unique' used
+ $this->hasColumn("address","string",150,"email|unique");
+ }
+}
+$session = Doctrine_Manager::getInstance()->openSession(new PDO("dsn","username","password"));
+$user = new User();
+$user->name = "this is an example of too long name";
+
+$user->save(); // throws a Doctrine_Validator_Exception
+
+$user->name = "valid name";
+$user->created = "not valid"; // not valid type
+$user->save(); // throws a Doctrine_Validator_Exception
+
+
+$user->created = time();
+$user->Email->address = "drink@.."; // not valid email address
+$user->save(); // throws a Doctrine_Validator_Exception
+
+$user->Email->address = "drink@drinkmore.info";
+$user->save(); // saved
+
+
+$user = $session->create("User");
+$user->Email->address = "drink@drinkmore.info"; // not unique!
+$user->save(); // throws a Doctrine_Validator_Exception
+?>
diff --git a/manual/codes/Basic Components - Collection - Accessing elements.php b/manual/codes/Basic Components - Collection - Accessing elements.php
new file mode 100644
index 000000000..47e8519b6
--- /dev/null
+++ b/manual/codes/Basic Components - Collection - Accessing elements.php
@@ -0,0 +1,15 @@
+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;
+?>
diff --git a/manual/codes/Basic Components - Collection - Adding new elements.php b/manual/codes/Basic Components - Collection - Adding new elements.php
new file mode 100644
index 000000000..dec8c7693
--- /dev/null
+++ b/manual/codes/Basic Components - Collection - Adding new elements.php
@@ -0,0 +1,8 @@
+findAll();
+
+print count($users); // 5
+
+$users[5]->name = "new user 1";
+$users[6]->name = "new user 2";
+?>
diff --git a/manual/codes/Basic Components - Collection - Deleting collection.php b/manual/codes/Basic Components - Collection - Deleting collection.php
new file mode 100644
index 000000000..d4f288459
--- /dev/null
+++ b/manual/codes/Basic Components - Collection - Deleting collection.php
@@ -0,0 +1,7 @@
+findBySql("name LIKE '%John%'");
+
+$users->delete();
+?>
diff --git a/manual/codes/Basic Components - Collection - Fetching strategies.php b/manual/codes/Basic Components - Collection - Fetching strategies.php
new file mode 100644
index 000000000..ee5cd03ef
--- /dev/null
+++ b/manual/codes/Basic Components - Collection - Fetching strategies.php
@@ -0,0 +1,52 @@
+getTable("User");
+
+$table->setAttribute(Doctrine::ATTR_FETCHMODE, Doctrine::FETCH_IMMEDIATE);
+
+$users = $table->findAll();
+
+// or
+
+$users = $session->query("FROM User-I"); // immediate collection
+
+foreach($users as $user) {
+ print $user->name;
+}
+
+
+$table->setAttribute(Doctrine::ATTR_FETCHMODE, Doctrine::FETCH_LAZY);
+
+$users = $table->findAll();
+
+// or
+
+$users = $session->query("FROM User-L"); // lazy collection
+
+foreach($users as $user) {
+ print $user->name;
+}
+
+$table->setAttribute(Doctrine::ATTR_FETCHMODE, Doctrine::FETCH_BATCH);
+
+$users = $table->findAll();
+
+// or
+
+$users = $session->query("FROM User-B"); // batch collection
+
+foreach($users as $user) {
+ print $user->name;
+}
+
+$table->setAttribute(Doctrine::ATTR_FETCHMODE, Doctrine::FETCH_OFFSET);
+
+$users = $table->findAll();
+
+// or
+
+$users = $session->query("FROM User-O"); // offset collection
+
+foreach($users as $user) {
+ print $user->name;
+}
+?>
diff --git a/manual/codes/Basic Components - Collection - Getting collection count.php b/manual/codes/Basic Components - Collection - Getting collection count.php
new file mode 100644
index 000000000..a2e88a5d1
--- /dev/null
+++ b/manual/codes/Basic Components - Collection - Getting collection count.php
@@ -0,0 +1,9 @@
+findAll();
+
+$users->count();
+
+// or
+
+count($users); // Doctrine_Collection implements Countable interface
+?>
diff --git a/manual/codes/Basic Components - Collection - Saving the collection.php b/manual/codes/Basic Components - Collection - Saving the collection.php
new file mode 100644
index 000000000..7ef541270
--- /dev/null
+++ b/manual/codes/Basic Components - Collection - Saving the collection.php
@@ -0,0 +1,9 @@
+findAll();
+
+$users[0]->name = "Jack Daniels";
+
+$users[1]->name = "John Locke";
+
+$users->save();
+?>
diff --git a/manual/codes/Basic Components - Manager - Managing sessions.php b/manual/codes/Basic Components - Manager - Managing sessions.php
new file mode 100644
index 000000000..ac67cb878
--- /dev/null
+++ b/manual/codes/Basic Components - Manager - Managing sessions.php
@@ -0,0 +1,25 @@
+openSession(new PDO("dsn","username","password"), "session 1");
+
+// open second session
+
+$session2 = $manager->openSession(new PDO("dsn2","username2","password2"), "session 2");
+
+$manager->getCurrentSession(); // $session2
+
+$manager->setCurrentSession("session 1");
+
+$manager->getCurrentSession(); // $session
+
+// iterating through sessions
+
+foreach($manager as $session) {
+
+}
+?>
diff --git a/manual/codes/Basic Components - Manager - Opening a new session.php b/manual/codes/Basic Components - Manager - Opening a new session.php
new file mode 100644
index 000000000..8e774cb8e
--- /dev/null
+++ b/manual/codes/Basic Components - Manager - Opening a new session.php
@@ -0,0 +1,18 @@
+openSession();
+
+// or if you want to use Doctrine Doctrine_DB and its
+// performance monitoring capabilities
+
+$dsn = "schema://username:password@dsn/dbname";
+$dbh = Doctrine_DB::getConnection($dsn);
+$session = $manager->openSession();
+?>
diff --git a/manual/codes/Basic Components - Query - Bound parameters.php b/manual/codes/Basic Components - Query - Bound parameters.php
new file mode 100644
index 000000000..9bb9a97af
--- /dev/null
+++ b/manual/codes/Basic Components - Query - Bound parameters.php
@@ -0,0 +1,6 @@
+from("User")
+ ->where("User.name = ?");
+
+$query->execute(array('Jack Daniels'));
+?>
diff --git a/manual/codes/Basic Components - Query - FROM - selecting tables.php b/manual/codes/Basic Components - Query - FROM - selecting tables.php
new file mode 100644
index 000000000..ebaafeda8
--- /dev/null
+++ b/manual/codes/Basic Components - Query - FROM - selecting tables.php
@@ -0,0 +1,15 @@
+query("FROM User");
+
+// find all groups
+
+$coll = $session->query("FROM Group");
+
+// find all users and user emails
+
+$coll = $session->query("FROM User, User.Email");
+
+?>
diff --git a/manual/codes/Basic Components - Query - Fetching strategies.php b/manual/codes/Basic Components - Query - Fetching strategies.php
new file mode 100644
index 000000000..efdd801c4
--- /dev/null
+++ b/manual/codes/Basic Components - Query - Fetching strategies.php
@@ -0,0 +1,25 @@
+query("FROM User-I");
+
+// or
+
+$coll = $session->query("FROM User-IMMEDIATE");
+
+// select all users and load the data in batches
+
+$coll = $session->query("FROM User-B");
+
+// or
+
+$coll = $session->query("FROM User-BATCH");
+
+// select all user and use lazy fetching
+
+$coll = $session->query("FROM User-L");
+
+// or
+
+$coll = $session->query("FROM User-LAZY");
+?>
diff --git a/manual/codes/Basic Components - Query - LIMIT and OFFSET - limiting the query results.php b/manual/codes/Basic Components - Query - LIMIT and OFFSET - limiting the query results.php
new file mode 100644
index 000000000..02edc71b6
--- /dev/null
+++ b/manual/codes/Basic Components - Query - LIMIT and OFFSET - limiting the query results.php
@@ -0,0 +1,11 @@
+query("FROM User, User.Email LIMIT 10");
+
+// find the first ten users starting from the user number 5
+
+$coll = $session->query("FROM User LIMIT 10 OFFSET 5");
+
+?>
diff --git a/manual/codes/Basic Components - Query - Lazy property fetching.php b/manual/codes/Basic Components - Query - Lazy property fetching.php
new file mode 100644
index 000000000..d9eba2558
--- /dev/null
+++ b/manual/codes/Basic Components - Query - Lazy property fetching.php
@@ -0,0 +1,6 @@
+query("FROM User(id, name)");
+?>
diff --git a/manual/codes/Basic Components - Query - Method overloading.php b/manual/codes/Basic Components - Query - Method overloading.php
new file mode 100644
index 000000000..33d39bad7
--- /dev/null
+++ b/manual/codes/Basic Components - Query - Method overloading.php
@@ -0,0 +1,12 @@
+openSession(new PDO("dsn","username","password"));
+
+$query = new Doctrine_Query($session);
+
+$query->from("User-b")
+ ->where("User.name LIKE 'Jack%'")
+ ->orderby("User.created");
+ ->limit(5);
+
+$users = $query->execute();
+?>
diff --git a/manual/codes/Basic Components - Query - ORDER BY - sorting query results.php b/manual/codes/Basic Components - Query - ORDER BY - sorting query results.php
new file mode 100644
index 000000000..e3650e7ce
--- /dev/null
+++ b/manual/codes/Basic Components - Query - ORDER BY - sorting query results.php
@@ -0,0 +1,21 @@
+query("FROM User ORDER BY User.name DESC");
+
+// find all users sort by name ascending
+
+$coll = $session->query("FROM User ORDER BY User.name ASC");
+
+// or
+
+$coll = $session->query("FROM User ORDER BY User.name");
+
+// find all users and their emails, sort by email address
+
+$coll = $session->query("FROM User, User.Email ORDER BY User.Email.address");
+
+// find all users and their emails, sort by user name and email address
+
+$coll = $session->query("FROM User, User.Email ORDER BY User.name, User.Email.address");
+?>
diff --git a/manual/codes/Basic Components - Query - Relation operators.php b/manual/codes/Basic Components - Query - Relation operators.php
new file mode 100644
index 000000000..977624130
--- /dev/null
+++ b/manual/codes/Basic Components - Query - Relation operators.php
@@ -0,0 +1,15 @@
+from("User:Email");
+
+$query->execute();
+
+// executed SQL query:
+// SELECT ... FROM user INNER JOIN email ON ...
+
+$query->from("User.Email");
+
+$query->execute();
+
+// executed SQL query:
+// SELECT ... FROM user LEFT JOIN email ON ...
+?>
diff --git a/manual/codes/Basic Components - Query - WHERE - setting query conditions.php b/manual/codes/Basic Components - Query - WHERE - setting query conditions.php
new file mode 100644
index 000000000..0b48cc55e
--- /dev/null
+++ b/manual/codes/Basic Components - Query - WHERE - setting query conditions.php
@@ -0,0 +1,27 @@
+query("FROM Group WHERE Group.id > 10");
+
+// find all users where users where user name matches a regular expression,
+// REGEXP keyword must be supported by the underlying database
+
+$coll = $session->query("FROM User WHERE User.name REGEXP '[ad]'");
+
+// find all users and their associated emails where SOME of the users phonenumbers
+// (the association between user and phonenumber tables is Many-To-Many) starts with 123
+
+$coll = $session->query("FROM User, User.Email WHERE User.Phonenumber.phonenumber LIKE '123%'");
+
+// multiple conditions
+
+$coll = $session->query("FROM User WHERE User.name LIKE '%Jack%' && User.Email.address LIKE '%@drinkmore.info'");
+
+// nesting conditions
+
+$coll = $session->query("FROM User WHERE (User.name LIKE '%Jack%' || User.name LIKE '%John%') && User.Email.address LIKE '%@drinkmore.info'");
+
+?>
diff --git a/manual/codes/Basic Components - RawSql - Method overloading.php b/manual/codes/Basic Components - RawSql - Method overloading.php
new file mode 100644
index 000000000..8b49c705b
--- /dev/null
+++ b/manual/codes/Basic Components - RawSql - Method overloading.php
@@ -0,0 +1,10 @@
+select('{entity.name}')
+ ->from('entity');
+
+$query->addComponent("entity", "User");
+
+$coll = $query->execute();
+?>
diff --git a/manual/codes/Basic Components - RawSql - Using SQL.php b/manual/codes/Basic Components - RawSql - Using SQL.php
new file mode 100644
index 000000000..973ec6744
--- /dev/null
+++ b/manual/codes/Basic Components - RawSql - Using SQL.php
@@ -0,0 +1,7 @@
+parseQuery("SELECT {entity.name} FROM entity");
+
+$entities = $query->execute();
+?>
diff --git a/manual/codes/Basic Components - Record - Accessing properties.php b/manual/codes/Basic Components - Record - Accessing properties.php
new file mode 100644
index 000000000..f02e1d2c7
--- /dev/null
+++ b/manual/codes/Basic Components - Record - Accessing properties.php
@@ -0,0 +1,21 @@
+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) {
+
+}
+?>
diff --git a/manual/codes/Basic Components - Record - Creating new records.php b/manual/codes/Basic Components - Record - Creating new records.php
new file mode 100644
index 000000000..97bee6142
--- /dev/null
+++ b/manual/codes/Basic Components - Record - Creating new records.php
@@ -0,0 +1,20 @@
+create("User");
+
+// alternative way:
+
+$table = $session->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();
+?>
diff --git a/manual/codes/Basic Components - Record - Deleting records.php b/manual/codes/Basic Components - Record - Deleting records.php
new file mode 100644
index 000000000..5cf6a1630
--- /dev/null
+++ b/manual/codes/Basic Components - Record - Deleting records.php
@@ -0,0 +1,20 @@
+getTable("User");
+
+try {
+ $user = $table->find(2);
+} catch(Doctrine_Find_Exception $e) {
+ print "Couldn't find user";
+}
+
+// deletes user and all related composite objects
+
+$user->delete();
+
+
+$users = $table->findAll();
+
+
+// delete all users and their related composite objects
+$users->delete();
+?>
diff --git a/manual/codes/Basic Components - Record - Getting object copy.php b/manual/codes/Basic Components - Record - Getting object copy.php
new file mode 100644
index 000000000..e5d527d75
--- /dev/null
+++ b/manual/codes/Basic Components - Record - Getting object copy.php
@@ -0,0 +1,3 @@
+copy();
+?>
diff --git a/manual/codes/Basic Components - Record - Getting record state.php b/manual/codes/Basic Components - Record - Getting record state.php
new file mode 100644
index 000000000..21148fca2
--- /dev/null
+++ b/manual/codes/Basic Components - Record - Getting record state.php
@@ -0,0 +1,31 @@
+getState();
+
+switch($state):
+ case Doctrine_Record::STATE_PROXY:
+ // data access object is in proxy state,
+ // meaning its persistent but not all of its properties are
+ // loaded from the database
+ break;
+ case Doctrine_Record::STATE_TCLEAN:
+ // data access object is transient clean,
+ // meaning its transient and
+ // none of its properties are changed
+ break;
+ case Doctrine_Record::STATE_TDIRTY:
+ // data access object is transient dirty,
+ // meaning its transient and
+ // some of its properties are changed
+ break;
+ case Doctrine_Record::STATE_DIRTY:
+ // data access object is dirty,
+ // meaning its persistent and
+ // some of its properties are changed
+ break;
+ case Doctrine_Record::STATE_CLEAN:
+ // data access object is clean,
+ // meaning its persistent and
+ // none of its properties are changed
+ break;
+endswitch;
+?>
diff --git a/manual/codes/Basic Components - Record - Retrieving existing records.php b/manual/codes/Basic Components - Record - Retrieving existing records.php
new file mode 100644
index 000000000..cee6a20fb
--- /dev/null
+++ b/manual/codes/Basic Components - Record - Retrieving existing records.php
@@ -0,0 +1,24 @@
+getTable("User");
+
+// find by primary key
+try {
+ $user = $table->find(2);
+} catch(Doctrine_Find_Exception $e) {
+ print "Couldn't find user";
+}
+
+// get all users
+foreach($table->findAll() as $user) {
+ print $user->name;
+}
+
+// finding by sql
+foreach($table->findBySql("name LIKE '%John%'") as $user) {
+ print $user->created;
+}
+
+// finding objects with DQL
+
+$users = $session->query("FROM User WHERE User.name LIKE '%John%'");
+?>
diff --git a/manual/codes/Basic Components - Record - Serializing.php b/manual/codes/Basic Components - Record - Serializing.php
new file mode 100644
index 000000000..1ae2e9933
--- /dev/null
+++ b/manual/codes/Basic Components - Record - Serializing.php
@@ -0,0 +1,5 @@
+
diff --git a/manual/codes/Basic Components - Record - Updating records.php b/manual/codes/Basic Components - Record - Updating records.php
new file mode 100644
index 000000000..b5d318364
--- /dev/null
+++ b/manual/codes/Basic Components - Record - Updating records.php
@@ -0,0 +1,13 @@
+getTable("User");
+
+try {
+ $user = $table->find(2);
+} catch(Doctrine_Find_Exception $e) {
+ print "Couldn't find user";
+}
+
+$user->name = "Jack Daniels";
+
+$user->save();
+?>
diff --git a/manual/codes/Basic Components - Session - Flushing the session.php b/manual/codes/Basic Components - Session - Flushing the session.php
new file mode 100644
index 000000000..1f56b7e1f
--- /dev/null
+++ b/manual/codes/Basic Components - Session - Flushing the session.php
@@ -0,0 +1,11 @@
+name = "Jack";
+
+$group = $session->create("Group");
+$group->name = "Drinking Club";
+
+// saves all the changed objects into database
+
+$session->flush();
+?>
diff --git a/manual/codes/Basic Components - Session - Getting a table object.php b/manual/codes/Basic Components - Session - Getting a table object.php
new file mode 100644
index 000000000..1f5e21983
--- /dev/null
+++ b/manual/codes/Basic Components - Session - Getting a table object.php
@@ -0,0 +1,11 @@
+openSession(new PDO("dsn","username","password"));
+
+// getting a table object
+
+$table = $session->getTable("User");
+?>
diff --git a/manual/codes/Basic Components - Session - Getting session state.php b/manual/codes/Basic Components - Session - Getting session state.php
new file mode 100644
index 000000000..b1589f815
--- /dev/null
+++ b/manual/codes/Basic Components - Session - Getting session state.php
@@ -0,0 +1,16 @@
+getState())
+ case Doctrine_Session::STATE_ACTIVE:
+ // session open and zero open transactions
+ break;
+ case Doctrine_Session::STATE_ACTIVE:
+ // one open transaction
+ break;
+ case Doctrine_Session::STATE_BUSY:
+ // multiple open transactions
+ break;
+ case Doctrine_Session::STATE_CLOSED:
+ // session closed
+ break;
+endswitch;
+?>
diff --git a/manual/codes/Basic Components - Session - Querying the database.php b/manual/codes/Basic Components - Session - Querying the database.php
new file mode 100644
index 000000000..cfea02b91
--- /dev/null
+++ b/manual/codes/Basic Components - Session - Querying the database.php
@@ -0,0 +1,10 @@
+query("FROM User");
+
+// select all users where user email is jackdaniels@drinkmore.info
+
+$session->query("FROM User WHERE User.Email.address = 'jackdaniels@drinkmore.info'");
+?>
diff --git a/manual/codes/Basic Components - Table - Custom finders.php b/manual/codes/Basic Components - Table - Custom finders.php
new file mode 100644
index 000000000..50e0f6732
--- /dev/null
+++ b/manual/codes/Basic Components - Table - Custom finders.php
@@ -0,0 +1,22 @@
+getSession()->query("FROM User WHERE name LIKE '%$name%'");
+ }
+}
+class User extends Doctrine_Record { }
+
+$session = Doctrine_Manager::getInstance()->openSession(new PDO("dsn","username","password"));
+
+// doctrine will now check if a class called UserTable exists and if it inherits Doctrine_Table
+
+$table = $session->getTable("User");
+
+print get_class($table); // UserTable
+
+$users = $table->findByName("Jack");
+
+?>
diff --git a/manual/codes/Basic Components - Table - Custom table classes.php b/manual/codes/Basic Components - Table - Custom table classes.php
new file mode 100644
index 000000000..138d5a445
--- /dev/null
+++ b/manual/codes/Basic Components - Table - Custom table classes.php
@@ -0,0 +1,13 @@
+
+
+
diff --git a/manual/codes/Basic Components - Table - Finder methods.php b/manual/codes/Basic Components - Table - Finder methods.php
new file mode 100644
index 000000000..a78103611
--- /dev/null
+++ b/manual/codes/Basic Components - Table - Finder methods.php
@@ -0,0 +1,20 @@
+getTable("User");
+
+// find by primary key
+try {
+ $user = $table->find(2);
+} catch(Doctrine_Find_Exception $e) {
+ print "Couldn't find user";
+}
+
+// get all users
+foreach($table->findAll() as $user) {
+ print $user->name;
+}
+
+// finding by sql
+foreach($table->findBySql("name LIKE '%John%'") as $user) {
+ print $user->created;
+}
+?>
diff --git a/manual/codes/Basic Components - Table - Getting table information.php b/manual/codes/Basic Components - Table - Getting table information.php
new file mode 100644
index 000000000..ed077cdda
--- /dev/null
+++ b/manual/codes/Basic Components - Table - Getting table information.php
@@ -0,0 +1,11 @@
+getTable('User');
+
+// getting column names
+
+$names = $table->getColumnNames();
+
+// getting column information
+
+$columns = $table->getColumns();
+?>
diff --git a/manual/codes/Configuration - Custom getters and setters.php b/manual/codes/Configuration - Custom getters and setters.php
new file mode 100644
index 000000000..0cf8a244f
--- /dev/null
+++ b/manual/codes/Configuration - Custom getters and setters.php
@@ -0,0 +1,21 @@
+isValidName($name))
+ $this->set("name",$name);
+ }
+ public function getName() {
+ return $this->get("name");
+ }
+}
+?>
diff --git a/manual/codes/Configuration - Custom primary key column.php b/manual/codes/Configuration - Custom primary key column.php
new file mode 100644
index 000000000..1535df124
--- /dev/null
+++ b/manual/codes/Configuration - Custom primary key column.php
@@ -0,0 +1,9 @@
+setPrimaryKeyColumn("group_id");
+ }
+}
+?>
diff --git a/manual/codes/Configuration - Setting attributes - Batch size.php b/manual/codes/Configuration - Setting attributes - Batch size.php
new file mode 100644
index 000000000..d7640f7e3
--- /dev/null
+++ b/manual/codes/Configuration - Setting attributes - Batch size.php
@@ -0,0 +1,5 @@
+setAttribute(Doctrine::ATTR_BATCH_SIZE, 7);
+?>
diff --git a/manual/codes/Configuration - Setting attributes - Event listener.php b/manual/codes/Configuration - Setting attributes - Event listener.php
new file mode 100644
index 000000000..9d308e058
--- /dev/null
+++ b/manual/codes/Configuration - Setting attributes - Event listener.php
@@ -0,0 +1,5 @@
+setAttribute(Doctrine::ATTR_LISTENER, new MyListener());
+?>
diff --git a/manual/codes/Configuration - Setting attributes - Fetching strategy.php b/manual/codes/Configuration - Setting attributes - Fetching strategy.php
new file mode 100644
index 000000000..576f6aa7f
--- /dev/null
+++ b/manual/codes/Configuration - Setting attributes - Fetching strategy.php
@@ -0,0 +1,5 @@
+setAttribute(Doctrine::ATTR_FETCHMODE, Doctrine::FETCH_LAZY);
+?>
diff --git a/manual/codes/Configuration - Setting attributes - Offset collection limit.php b/manual/codes/Configuration - Setting attributes - Offset collection limit.php
new file mode 100644
index 000000000..1f3c02bae
--- /dev/null
+++ b/manual/codes/Configuration - Setting attributes - Offset collection limit.php
@@ -0,0 +1,5 @@
+setAttribute(Doctrine::ATTR_COLL_LIMIT, 10);
+?>
diff --git a/manual/codes/Configuration - Setting attributes - Session lockmode.php b/manual/codes/Configuration - Setting attributes - Session lockmode.php
new file mode 100644
index 000000000..e92f4e233
--- /dev/null
+++ b/manual/codes/Configuration - Setting attributes - Session lockmode.php
@@ -0,0 +1,5 @@
+setAttribute(Doctrine::ATTR_LOCKMODE, Doctrine::LOCK_PESSIMISTIC);
+?>
diff --git a/manual/codes/Configuration - Setting attributes - Table creation.php b/manual/codes/Configuration - Setting attributes - Table creation.php
new file mode 100644
index 000000000..87b2c645a
--- /dev/null
+++ b/manual/codes/Configuration - Setting attributes - Table creation.php
@@ -0,0 +1,5 @@
+setAttribute(Doctrine::ATTR_CREATE_TABLES, false);
+?>
diff --git a/manual/codes/Configuration - Setting attributes - Validation.php b/manual/codes/Configuration - Setting attributes - Validation.php
new file mode 100644
index 000000000..008d2b7b9
--- /dev/null
+++ b/manual/codes/Configuration - Setting attributes - Validation.php
@@ -0,0 +1,5 @@
+setAttribute(Doctrine::ATTR_VLD, true);
+?>
diff --git a/manual/codes/Configuration - Setting default eventlistener.php b/manual/codes/Configuration - Setting default eventlistener.php
new file mode 100644
index 000000000..a3cb4920b
--- /dev/null
+++ b/manual/codes/Configuration - Setting default eventlistener.php
@@ -0,0 +1,10 @@
+setAttribute(Doctrine::ATTR_LISTENER,new MyListener());
+ }
+ public function setTableDefinition() {
+ $this->hasColumn("address","string",150,"email|unique");
+ }
+}
+?>
diff --git a/manual/codes/Configuration - Setting default fetchmode.php b/manual/codes/Configuration - Setting default fetchmode.php
new file mode 100644
index 000000000..bee2e999c
--- /dev/null
+++ b/manual/codes/Configuration - Setting default fetchmode.php
@@ -0,0 +1,11 @@
+setAttribute(Doctrine::ATTR_FETCHMODE,Doctrine::FETCH_IMMEDIATE);
+ }
+}
+?>
diff --git a/manual/codes/Configuration - Using sequences.php b/manual/codes/Configuration - Using sequences.php
new file mode 100644
index 000000000..16ed99419
--- /dev/null
+++ b/manual/codes/Configuration - Using sequences.php
@@ -0,0 +1,9 @@
+setSequenceName("user_seq");
+ }
+}
+?>
diff --git a/manual/codes/Database operations - Limit and offset.php b/manual/codes/Database operations - Limit and offset.php
new file mode 100644
index 000000000..b770a6f6b
--- /dev/null
+++ b/manual/codes/Database operations - Limit and offset.php
@@ -0,0 +1,7 @@
+openSession(new PDO("dsn","username","password"));
+
+// select first ten rows starting from the row 20
+
+$sess->select("select * from user",10,20);
+?>
diff --git a/manual/codes/Database operations - Nested transactions.php b/manual/codes/Database operations - Nested transactions.php
new file mode 100644
index 000000000..22cf0924e
--- /dev/null
+++ b/manual/codes/Database operations - Nested transactions.php
@@ -0,0 +1,17 @@
+beginTransaction();
+
+ $user->save();
+
+ $session->beginTransaction();
+ $group->save();
+ $email->save();
+
+ $session->commit();
+
+ $session->commit();
+} catch(Exception $e) {
+ $session->rollback();
+}
+?>
diff --git a/manual/codes/Database operations - Query logging.php b/manual/codes/Database operations - Query logging.php
new file mode 100644
index 000000000..2694b96c9
--- /dev/null
+++ b/manual/codes/Database operations - Query logging.php
@@ -0,0 +1,15 @@
+getDBH();
+
+$times = $dbh->getExecTimes();
+
+// print all executed queries and their execution times
+
+foreach($dbh->getQueries() as $index => $query) {
+ print $query." ".$times[$index];
+}
+
+?>
diff --git a/manual/codes/Database operations - Sequences.php b/manual/codes/Database operations - Sequences.php
new file mode 100644
index 000000000..b671dad1d
--- /dev/null
+++ b/manual/codes/Database operations - Sequences.php
@@ -0,0 +1,7 @@
+openSession(new PDO("dsn","username","password"));
+
+// gets the next ID from a sequence
+
+$sess->getNextID($sequence);
+?>
diff --git a/manual/codes/Database operations - Transactions.php b/manual/codes/Database operations - Transactions.php
new file mode 100644
index 000000000..46d515774
--- /dev/null
+++ b/manual/codes/Database operations - Transactions.php
@@ -0,0 +1,14 @@
+openSession(new PDO("dsn","username","password"));
+try {
+$sess->beginTransaction();
+
+ // some database operations
+
+$sess->commit();
+
+} catch(Exception $e) {
+ $sess->rollback();
+}
+
+?>
diff --git a/manual/codes/Getting started - Installation.php b/manual/codes/Getting started - Installation.php
new file mode 100644
index 000000000..374c50e47
--- /dev/null
+++ b/manual/codes/Getting started - Installation.php
@@ -0,0 +1,13 @@
+
diff --git a/manual/codes/Getting started - Setting table definition.php b/manual/codes/Getting started - Setting table definition.php
new file mode 100644
index 000000000..23b9057be
--- /dev/null
+++ b/manual/codes/Getting started - Setting table definition.php
@@ -0,0 +1,15 @@
+hasColumn("address","string",200,"email|unique");
+ }
+}
+?>
diff --git a/manual/codes/Getting started - Starting new project.php b/manual/codes/Getting started - Starting new project.php
new file mode 100644
index 000000000..5cda71396
--- /dev/null
+++ b/manual/codes/Getting started - Starting new project.php
@@ -0,0 +1,13 @@
+hasColumn("name","string",30);
+ $this->hasColumn("username","string",20);
+ $this->hasColumn("password","string",16);
+ $this->hasColumn("created","integer",11);
+ }
+}
+?>
diff --git a/manual/codes/Mapping object relations - Dealing with relations - Creating related records.php b/manual/codes/Mapping object relations - Dealing with relations - Creating related records.php
new file mode 100644
index 000000000..03c7edcc9
--- /dev/null
+++ b/manual/codes/Mapping object relations - Dealing with relations - Creating related records.php
@@ -0,0 +1,14 @@
+Email;
+
+$email->address = "jackdaniels@drinkmore.info";
+
+$user->save();
+
+// alternative:
+
+$user->Email->address = "jackdaniels@drinkmore.info";
+
+$user->save();
+?>
diff --git a/manual/codes/Mapping object relations - Dealing with relations - Deleting related records.php b/manual/codes/Mapping object relations - Dealing with relations - Deleting related records.php
new file mode 100644
index 000000000..e936c3cff
--- /dev/null
+++ b/manual/codes/Mapping object relations - Dealing with relations - Deleting related records.php
@@ -0,0 +1,9 @@
+Email->delete();
+
+$user->Phonenumber[3]->delete();
+
+// deleting user and all related objects:
+
+$user->delete();
+?>
diff --git a/manual/codes/Mapping object relations - Dealing with relations - Retrieving related records.php b/manual/codes/Mapping object relations - Dealing with relations - Retrieving related records.php
new file mode 100644
index 000000000..d2d9d7511
--- /dev/null
+++ b/manual/codes/Mapping object relations - Dealing with relations - Retrieving related records.php
@@ -0,0 +1,7 @@
+Email["address"];
+
+print $user->Phonenumber[0]->phonenumber;
+
+print $user->Group[0]->get("name");
+?>
diff --git a/manual/codes/Mapping object relations - Dealing with relations - Updating related records.php b/manual/codes/Mapping object relations - Dealing with relations - Updating related records.php
new file mode 100644
index 000000000..6df14cb1a
--- /dev/null
+++ b/manual/codes/Mapping object relations - Dealing with relations - Updating related records.php
@@ -0,0 +1,9 @@
+Email["address"] = "koskenkorva@drinkmore.info";
+
+$user->Phonenumber[0]->phonenumber = "123123";
+
+$user->save();
+
+// saves the email and phonenumber
+?>
diff --git a/manual/codes/Mapping object relations - Foreign key associations - One-to-Many, Many-to-One.php b/manual/codes/Mapping object relations - Foreign key associations - One-to-Many, Many-to-One.php
new file mode 100644
index 000000000..6b05ecb83
--- /dev/null
+++ b/manual/codes/Mapping object relations - Foreign key associations - One-to-Many, Many-to-One.php
@@ -0,0 +1,18 @@
+ownsMany("Phonenumber","Phonenumber.user_id");
+ }
+ public function setTableDefition() {
+ $this->hasColumn("name","string",50);
+ $this->hasColumn("loginname","string",20);
+ $this->hasColumn("password","string",16);
+ }
+}
+class Phonenumber extends Doctrine_Record {
+ public function setTableDefinition() {
+ $this->hasColumn("phonenumber","string",50);
+ $this->hasColumn("user_id","integer");
+ }
+}
+?>
diff --git a/manual/codes/Mapping object relations - Foreign key associations - One-to-One.php b/manual/codes/Mapping object relations - Foreign key associations - One-to-One.php
new file mode 100644
index 000000000..9b91f625a
--- /dev/null
+++ b/manual/codes/Mapping object relations - Foreign key associations - One-to-One.php
@@ -0,0 +1,28 @@
+hasOne("Address","Address.user_id");
+ $this->ownsOne("Email","User.email_id");
+ $this->ownsMany("Phonenumber","Phonenumber.user_id");
+ }
+ public function setTableDefition() {
+ $this->hasColumn("name","string",50);
+ $this->hasColumn("loginname","string",20);
+ $this->hasColumn("password","string",16);
+
+ // foreign key column for email ID
+ $this->hasColumn("email_id","integer");
+ }
+}
+class Email extends Doctrine_Record {
+ public function setTableDefinition() {
+ $this->hasColumn("address","string",150);
+ }
+}
+class Address extends Doctrine_Record {
+ public function setTableDefinition() {
+ $this->hasColumn("street","string",50);
+ $this->hasColumn("user_id","integer");
+ }
+}
+?>
diff --git a/manual/codes/Mapping object relations - Foreign key associations - Tree structure.php b/manual/codes/Mapping object relations - Foreign key associations - Tree structure.php
new file mode 100644
index 000000000..eacecde72
--- /dev/null
+++ b/manual/codes/Mapping object relations - Foreign key associations - Tree structure.php
@@ -0,0 +1,12 @@
+hasOne("Task as Parent","Task.parent_id");
+ $this->hasMany("Task as Subtask","Subtask.parent_id");
+ }
+ public function setTableDefinition() {
+ $this->hasColumn("name","string",100);
+ $this->hasColumn("parent_id","integer");
+ }
+}
+?>
diff --git a/manual/codes/Mapping object relations - Inheritance - Column aggregation.php b/manual/codes/Mapping object relations - Inheritance - Column aggregation.php
new file mode 100644
index 000000000..ad192fa37
--- /dev/null
+++ b/manual/codes/Mapping object relations - Inheritance - Column aggregation.php
@@ -0,0 +1,22 @@
+hasColumn("name","string",30);
+ $this->hasColumn("username","string",20);
+ $this->hasColumn("password","string",16);
+ $this->hasColumn("created","integer",11);
+ }
+}
+
+class User extends Entity {
+ public function setUp() {
+ $this->setInheritanceMap(array("type"=>1);
+ }
+}
+
+class Group extends Entity {
+ public function setUp() {
+ $this->setInheritanceMap(array("type"=>2);
+ }
+}
+?>
diff --git a/manual/codes/Mapping object relations - Inheritance - One table many classes.php b/manual/codes/Mapping object relations - Inheritance - One table many classes.php
new file mode 100644
index 000000000..543972352
--- /dev/null
+++ b/manual/codes/Mapping object relations - Inheritance - One table many classes.php
@@ -0,0 +1,14 @@
+hasColumn("name","string",30);
+ $this->hasColumn("username","string",20);
+ $this->hasColumn("password","string",16);
+ $this->hasColumn("created","integer",11);
+ }
+}
+
+class User extends Entity { }
+
+class Group extends Entity { }
+?>
diff --git a/manual/codes/Mapping object relations - Inheritance - One table one class.php b/manual/codes/Mapping object relations - Inheritance - One table one class.php
new file mode 100644
index 000000000..a93adeb92
--- /dev/null
+++ b/manual/codes/Mapping object relations - Inheritance - One table one class.php
@@ -0,0 +1,26 @@
+hasColumn("name","string",30);
+ $this->hasColumn("username","string",20);
+ $this->hasColumn("password","string",16);
+ $this->hasColumn("created","integer",11);
+ }
+}
+
+class User extends Entity {
+ public function setTableDefinition() {
+ // the following method call is needed in
+ // one-table-one-class inheritance
+ parent::setTableDefinition();
+ }
+}
+
+class Group extends Entity {
+ public function setTableDefinition() {
+ // the following method call is needed in
+ // one-table-one-class inheritance
+ parent::setTableDefinition();
+ }
+}
+?>
diff --git a/manual/codes/Mapping object relations - Join table associations - Many-to-Many.php b/manual/codes/Mapping object relations - Join table associations - Many-to-Many.php
new file mode 100644
index 000000000..82571d303
--- /dev/null
+++ b/manual/codes/Mapping object relations - Join table associations - Many-to-Many.php
@@ -0,0 +1,49 @@
+hasMany("Group","Groupuser.group_id");
+ }
+ public function setTableDefinition() {
+ $this->hasColumn("name","string",30);
+ }
+}
+
+class Group extends Doctrine_Record {
+ public function setUp() {
+ $this->hasMany("User","Groupuser.user_id");
+ }
+ public function setTableDefinition() {
+ $this->hasColumn("name","string",30);
+ }
+}
+
+class Groupuser extends Doctrine_Record {
+ public function setTableDefinition() {
+ $this->hasColumn("user_id","integer");
+ $this->hasColumn("group_id","integer");
+ }
+}
+
+
+$user = new User();
+
+// add two groups
+$user->Group[0]->name = "First Group";
+
+$user->Group[1]->name = "Second Group";
+
+// save changes into database
+$user->save();
+
+$groups = new Doctrine_Collection($session->getTable("Group"));
+
+$groups[0]->name = "Third Group";
+
+$groups[1]->name = "Fourth Group";
+
+$user->Group[2] = $groups[0];
+// $user will now have 3 groups
+
+$user->Group = $groups;
+// $user will now have two groups 'Third Group' and 'Fourth Group'
+?>
diff --git a/manual/codes/Mapping object relations - Join table associations - Self-referencing.php b/manual/codes/Mapping object relations - Join table associations - Self-referencing.php
new file mode 100644
index 000000000..57fdcd0c3
--- /dev/null
+++ b/manual/codes/Mapping object relations - Join table associations - Self-referencing.php
@@ -0,0 +1,16 @@
+hasMany("User as Friend","UserReference.user_id-user_id2");
+ }
+ public function setTableDefinition() {
+ $this->hasColumn("name","string",30);
+ }
+}
+class UserReference extends Doctrine_Record {
+ public function setTableDefinition() {
+ $this->hasColumn("user_id","integer");
+ $this->hasColumn("user_id2","integer");
+ }
+}
+?>
diff --git a/manual/codes/Mapping object relations - Relation aliases.php b/manual/codes/Mapping object relations - Relation aliases.php
new file mode 100644
index 000000000..e2ce4c75d
--- /dev/null
+++ b/manual/codes/Mapping object relations - Relation aliases.php
@@ -0,0 +1,26 @@
+hasColumn("name", "string", 100);
+ $this->hasColumn("description", "string", 5000);
+ }
+ public function setUp() {
+ // notice the 'as' keyword here
+ $this->ownsMany("Forum_Thread as Threads", "Forum_Thread.board_id");
+ }
+}
+
+class Forum_Thread extends Doctrine_Record {
+ public function setTableDefinition() {
+ $this->hasColumn("board_id", "integer", 10);
+ $this->hasColumn("updated", "integer", 10);
+ $this->hasColumn("closed", "integer", 1);
+ }
+ public function setUp() {
+ // notice the 'as' keyword here
+ $this->hasOne("Forum_Board as Board", "Forum_Thread.board_id");
+ }
+}
+$board = new Board();
+$board->Threads[0]->updated = time();
+?>
diff --git a/manual/codes/Performance - Internal optimizations - DELETE.php b/manual/codes/Performance - Internal optimizations - DELETE.php
new file mode 100644
index 000000000..2276fe51a
--- /dev/null
+++ b/manual/codes/Performance - Internal optimizations - DELETE.php
@@ -0,0 +1,24 @@
+delete();
+/**
+ * On session drivers other than mysql doctrine would now perform three queries
+ * regardless of how many users, emails and phonenumbers there are
+ *
+ * the queries would look something like:
+ * DELETE FROM entity WHERE entity.id IN (1,2,3, ... ,15)
+ * DELETE FROM phonenumber WHERE phonenumber.id IN (4,6,7,8)
+ * DELETE FROM email WHERE email.id IN (1,2, ... ,10)
+ *
+ * On mysql doctrine is EVEN SMARTER! Now it would perform only one query!
+ * the query would look like:
+ * DELETE entity, email, phonenumber FROM entity
+ * LEFT JOIN phonenumber ON entity.id = phonenumber.entity_id, email
+ * WHERE (entity.email_id = email.id) && (entity.id IN(4, 5, 6, 7, 8, 9, 10, 11)) && (entity.type = 0)
+ */
+
+
+?>
diff --git a/manual/codes/Performance - Internal optimizations - INSERT.php b/manual/codes/Performance - Internal optimizations - INSERT.php
new file mode 100644
index 000000000..43a9958b0
--- /dev/null
+++ b/manual/codes/Performance - Internal optimizations - INSERT.php
@@ -0,0 +1,28 @@
+save();
+/**
+ * now doctrine would perform prepared queries in the following order:
+ *
+ * first the emails since every user needs to get the primary key of their newly created email
+ * INSERT INTO email (address) VALUES (:address)
+ * INSERT INTO email (address) VALUES (:address)
+ * INSERT INTO email (address) VALUES (:address)
+ *
+ * then the users
+ * INSERT INTO entity (name,email_id) VALUES (:name,:email_id)
+ * INSERT INTO entity (name,email_id) VALUES (:name,:email_id)
+ * INSERT INTO entity (name,email_id) VALUES (:name,:email_id)
+ *
+ * and at last the phonenumbers since they need the primary keys of the newly created users
+ * INSERT INTO phonenumber (phonenumber,entity_id) VALUES (:phonenumber,:entity_id)
+ * INSERT INTO phonenumber (phonenumber,entity_id) VALUES (:phonenumber,:entity_id)
+ * INSERT INTO phonenumber (phonenumber,entity_id) VALUES (:phonenumber,:entity_id)
+ * INSERT INTO phonenumber (phonenumber,entity_id) VALUES (:phonenumber,:entity_id)
+ * INSERT INTO phonenumber (phonenumber,entity_id) VALUES (:phonenumber,:entity_id)
+ *
+ * These operations are considerably fast, since many databases perform multiple
+ * prepared queries very rapidly
+ */
+?>
diff --git a/manual/codes/Real world examples - Forum application.php b/manual/codes/Real world examples - Forum application.php
new file mode 100644
index 000000000..dbe0fe77e
--- /dev/null
+++ b/manual/codes/Real world examples - Forum application.php
@@ -0,0 +1,52 @@
+hasColumn("root_category_id", "integer", 10);
+ $this->hasColumn("parent_category_id", "integer", 10);
+ $this->hasColumn("name", "string", 50);
+ $this->hasColumn("description", "string", 99999);
+ }
+ public function setUp() {
+ $this->hasMany("Forum_Category as Subcategory", "Subcategory.parent_category_id");
+ $this->hasOne("Forum_Category as Rootcategory", "Forum_Category.root_category_id");
+ }
+}
+class Forum_Board extends Doctrine_Record {
+ public function setTableDefinition() {
+ $this->hasColumn("category_id", "integer", 10);
+ $this->hasColumn("name", "string", 100);
+ $this->hasColumn("description", "string", 5000);
+ }
+ public function setUp() {
+ $this->hasOne("Forum_Category as Category", "Forum_Board.category_id");
+ $this->ownsMany("Forum_Thread as Threads", "Forum_Thread.board_id");
+ }
+}
+
+class Forum_Entry extends Doctrine_Record {
+ public function setTableDefinition() {
+ $this->hasColumn("author", "string", 50);
+ $this->hasColumn("topic", "string", 100);
+ $this->hasColumn("message", "string", 99999);
+ $this->hasColumn("parent_entry_id", "integer", 10);
+ $this->hasColumn("thread_id", "integer", 10);
+ $this->hasColumn("date", "integer", 10);
+ }
+ public function setUp() {
+ $this->hasOne("Forum_Entry as Parent", "Forum_Entry.parent_entry_id");
+ $this->hasOne("Forum_Thread as Thread", "Forum_Entry.thread_id");
+ }
+}
+
+class Forum_Thread extends Doctrine_Record {
+ public function setTableDefinition() {
+ $this->hasColumn("board_id", "integer", 10);
+ $this->hasColumn("updated", "integer", 10);
+ $this->hasColumn("closed", "integer", 1);
+ }
+ public function setUp() {
+ $this->hasOne("Forum_Board as Board", "Forum_Thread.board_id");
+ $this->ownsMany("Forum_Entry as Entries", "Forum_Entry.thread_id");
+ }
+}
+?>
diff --git a/manual/codes/Real world examples - User management system.php b/manual/codes/Real world examples - User management system.php
new file mode 100644
index 000000000..de04d2763
--- /dev/null
+++ b/manual/codes/Real world examples - User management system.php
@@ -0,0 +1,79 @@
+ownsOne("Email","Entity.email_id");
+ $this->ownsMany("Phonenumber","Phonenumber.entity_id");
+ $this->setAttribute(Doctrine::ATTR_FETCHMODE,Doctrine::FETCH_BATCH);
+ $this->setAttribute(Doctrine::ATTR_LISTENER,new EntityListener());
+ }
+ public function setTableDefinition() {
+ $this->hasColumn("name","string",50);
+ $this->hasColumn("loginname","string",20);
+ $this->hasColumn("password","string",16);
+ $this->hasColumn("type","integer",1);
+ $this->hasColumn("created","integer",11);
+ $this->hasColumn("updated","integer",11);
+ $this->hasColumn("email_id","integer");
+ }
+ }
+}
+class Group extends Entity {
+ public function setUp() {
+ parent::setUp();
+ $this->hasMany("User","Groupuser.user_id");
+ $this->setInheritanceMap(array("type"=>1));
+ }
+}
+class User extends Entity {
+ public function setUp() {
+ parent::setUp();
+ $this->hasMany("Group","Groupuser.group_id");
+ $this->setInheritanceMap(array("type"=>0));
+ }
+}
+class Groupuser extends Doctrine_Record {
+ public function setTableDefinition() {
+ $this->hasColumn("group_id","integer");
+ $this->hasColumn("user_id","integer");
+ }
+}
+
+class Phonenumber extends Doctrine_Record {
+ public function setTableDefinition() {
+ $this->hasColumn("phonenumber","string",20);
+ $this->hasColumn("entity_id","integer");
+ }
+}
+class Email extends Doctrine_Record {
+ public function setTableDefinition() {
+ $this->hasColumn("address","string",150,"email|unique");
+ }
+}
+class EntityListener extends Doctrine_EventListener {
+ public function onPreUpdate(Doctrine_Record $record) {
+ $record->updated = time();
+ }
+ public function onPreInsert(Doctrine_Record $record) {
+ $record->created = time();
+ }
+}
+
+// USER MANAGEMENT SYSTEM IN ACTION:
+
+$manager = Doctrine_Manager::getInstance();
+
+$session = $manager->openSession(new PDO("DSN","username","password"));
+
+$user = new User();
+
+$user->name = "Jack Daniels";
+$user->Email->address = "jackdaniels@drinkmore.info";
+$user->Phonenumber[0]->phonenumber = "123 123";
+$user->Phonenumber[1]->phonenumber = "133 133";
+$user->save();
+
+$user->Group[0]->name = "beer lovers";
+$user->Group[0]->Email->address = "beerlovers@drinkmore.info";
+$user->Group[0]->save();
+
+?>
diff --git a/manual/codes/Runtime classes - Doctrine_Record.php b/manual/codes/Runtime classes - Doctrine_Record.php
new file mode 100644
index 000000000..0264a5ccf
--- /dev/null
+++ b/manual/codes/Runtime classes - Doctrine_Record.php
@@ -0,0 +1,40 @@
+find(2);
+
+// get state
+$state = $user->getState();
+
+print $user->name;
+
+print $user["name"];
+
+print $user->get("name");
+
+$user->name = "Jack Daniels";
+
+$user->set("name","Jack Daniels");
+
+// serialize record
+
+$serialized = serialize($user);
+
+$user = unserialize($serialized);
+
+// create a copy
+
+$copy = $user->copy();
+
+// get primary key
+
+$id = $user->getID();
+
+// print lots of useful info
+
+print $user;
+
+// save all the properties and composites
+$user->save();
+
+// delete this data access object and related objects
+$user->delete();
+?>
diff --git a/manual/codes/Runtime classes - Doctrine_Session.php b/manual/codes/Runtime classes - Doctrine_Session.php
new file mode 100644
index 000000000..b3c802415
--- /dev/null
+++ b/manual/codes/Runtime classes - Doctrine_Session.php
@@ -0,0 +1,30 @@
+openSession(Doctrine_DB::getConnection("schema://username:password@hostname/database"));
+
+// get session state:
+switch($sess):
+ case Doctrine_Session::STATE_BUSY:
+ // multiple open transactions
+ break;
+ case Doctrine_Session::STATE_ACTIVE:
+ // one open transaction
+ break;
+ case Doctrine_Session::STATE_CLOSED:
+ // closed state
+ break;
+ case Doctrine_Session::STATE_OPEN:
+ // open state and zero open transactions
+ break;
+endswitch;
+
+// getting database handler
+
+$dbh = $sess->getDBH();
+
+// flushing the session
+$sess->flush();
+
+
+// print lots of useful info about session:
+print $sess;
+?>
diff --git a/manual/codes/Transactions - Introduction.php b/manual/codes/Transactions - Introduction.php
new file mode 100644
index 000000000..5c02299e0
--- /dev/null
+++ b/manual/codes/Transactions - Introduction.php
@@ -0,0 +1,13 @@
+beginTransaction();
+
+$user = new User();
+$user->name = 'New user';
+$user->save();
+
+$user = $session->getTable('User')->find(5);
+$user->name = 'Modified user';
+$user->save();
+
+$session->commit(); // all the queries are executed here
+?>
diff --git a/manual/codes/Transactions - Nesting.php b/manual/codes/Transactions - Nesting.php
new file mode 100644
index 000000000..d64ec24c1
--- /dev/null
+++ b/manual/codes/Transactions - Nesting.php
@@ -0,0 +1,23 @@
+beginTransaction();
+
+ $user->save();
+
+ $group->save();
+
+ $session->commit();
+}
+
+try {
+ $session->beginTransaction();
+
+ saveUserAndGroup($session,$user,$group);
+ saveUserAndGroup($session,$user2,$group2);
+ saveUserAndGroup($session,$user3,$group3);
+
+ $session->commit();
+} catch(Doctrine_Exception $e) {
+ $session->rollback();
+}
+?>
diff --git a/manual/codes/Transactions - Unit of work.php b/manual/codes/Transactions - Unit of work.php
new file mode 100644
index 000000000..3946438bc
--- /dev/null
+++ b/manual/codes/Transactions - Unit of work.php
@@ -0,0 +1,18 @@
+beginTransaction();
+
+$user = new User();
+$user->name = 'New user';
+$user->save();
+
+$user = $session->getTable('User')->find(5);
+$user->name = 'Modified user';
+$user->save();
+
+
+$pending = $session->getInserts(); // an array containing one element
+
+$pending = $session->getUpdates(); // an array containing one element
+
+$session->commit(); // all the queries are executed here
+?>
diff --git a/manual/docs/Advanced components - Cache - Introduction.php b/manual/docs/Advanced components - Cache - Introduction.php
new file mode 100644
index 000000000..e7abab911
--- /dev/null
+++ b/manual/docs/Advanced components - Cache - Introduction.php
@@ -0,0 +1,2 @@
+Caching is one of the most influental things when it comes to performance tuning. Doctrine_Cache provides means for
+caching queries and for managing the cached queries.
diff --git a/manual/docs/Advanced components - Eventlisteners - Creating new listener.php b/manual/docs/Advanced components - Eventlisteners - Creating new listener.php
new file mode 100644
index 000000000..57f733cc0
--- /dev/null
+++ b/manual/docs/Advanced components - Eventlisteners - Creating new listener.php
@@ -0,0 +1 @@
+Creating a new listener is very easy. You can set the listener in global, session or factory level.
diff --git a/manual/docs/Advanced components - Eventlisteners - List of events.php b/manual/docs/Advanced components - Eventlisteners - List of events.php
new file mode 100644
index 000000000..95524cd93
--- /dev/null
+++ b/manual/docs/Advanced components - Eventlisteners - List of events.php
@@ -0,0 +1 @@
+Here is a list of availible events and their parameters:
diff --git a/manual/docs/Advanced components - Hook - Introduction.php b/manual/docs/Advanced components - Hook - Introduction.php
new file mode 100644
index 000000000..f5e977f91
--- /dev/null
+++ b/manual/docs/Advanced components - Hook - Introduction.php
@@ -0,0 +1,3 @@
+Many web applications have different kinds of lists. The lists may contain data from multiple components (= database tables) and
+they may have actions such as paging, sorting and setting conditions. Doctrine_Hook helps building these lists. It has a simple API for
+building search criteria forms as well as building a DQL query from the 'hooked' parameters.
diff --git a/manual/docs/Advanced components - Validators - Intruduction.php b/manual/docs/Advanced components - Validators - Intruduction.php
new file mode 100644
index 000000000..737dc2cd7
--- /dev/null
+++ b/manual/docs/Advanced components - Validators - Intruduction.php
@@ -0,0 +1,11 @@
+With Doctrine validators you can validate a whole transaction and get info of everything
+that went wrong. Some Doctrine validators also act as a database level constraints. For example
+adding a unique validator to column 'name' also adds a database level unique constraint into that
+column.
+
+Validators are added as a 4 argument for hasColumn() method. Validators should be separated
+by '|' mark. For example email|unique would validate a value using Doctrine_Validator_Email
+and Doctrine_Validator_Unique.
+
+Doctrine has many predefined validators (see chapter 13.3). If you wish to use
+custom validators just write *Validator classes and doctrine will automatically use them.
diff --git a/manual/docs/Advanced components - Validators - List of predefined validators.php b/manual/docs/Advanced components - Validators - List of predefined validators.php
new file mode 100644
index 000000000..186a75ef0
--- /dev/null
+++ b/manual/docs/Advanced components - Validators - List of predefined validators.php
@@ -0,0 +1,124 @@
+Here is a list of predefined validators. You cannot use these names for your custom validators.
+
+ name + | ++ arguments + | ++ task + | +
+ email + | ++ + | ++ Checks if value is valid email. + | +
+ notblank + | ++ + | ++ Checks if value is not blank. + | +
+ notnull + | ++ + | ++ Checks if value is not null. + | +
+ country + | ++ + | ++ Checks if value is valid country code. + | +
+ ip + | ++ + | ++ Checks if value is valid internet protocol address. + | +
+ htmlcolor + | ++ + | ++ Checks if value is valid html color. + | +
+ nospace + | ++ + | ++ Checks if value has no space chars. + | +
+ range + | ++ [min, max] + | ++ Checks if value is in range specified by arguments. + | +
+ unique + | ++ + | ++ Checks if value is unique in its database table. + | +
+ regexp + | ++ [expression] + | ++ Checks if value matches a given regexp. + | +
+InvalidKeyException + +Doctrine_Exception + +DQLException + +Doctrine_PrimaryKey_Exception thrown when Doctrine_Record is loaded and there is no primary key field + +Doctrine_Refresh_Exception thrown when Doctrine_Record is refreshed and the refreshed primary key doens't match the old one + +Doctrine_Find_Exception thrown when user tries to find a Doctrine_Record for given primary key and that object is not found + +Doctrine_Naming_Exception thrown when user defined Doctrine_Table is badly named + + +Doctrine_Session_Exception thrown when user tries to get the current + session and there are no open sessions + +Doctrine_Table_Exception thrown when user tries to initialize a new instance of Doctrine_Table, + while there already exists an instance of that factory + +Doctrine_Mapping_Exception thrown when user tries to get a foreign key object but the mapping is not done right +diff --git a/manual/docs/Cache - Configuration.php b/manual/docs/Cache - Configuration.php new file mode 100644 index 000000000..12b5d869b --- /dev/null +++ b/manual/docs/Cache - Configuration.php @@ -0,0 +1,26 @@ +There are couple of availible Cache attributes on Doctrine: +
+1. Every Doctrine_Table has its own cache directory. The default is cache/componentname/. All the cache files are saved into that directory.
+The format of each cache file is [primarykey].cache.
+ +2. When retrieving records from the database doctrine always tries to hit the cache first. + +3. If a record (Doctrine_Record) is retrieved from database or inserted into database it will be saved into cache. + +4. When a Data Access Object is deleted or updated it will be deleted from the cache + |
+
+1. Every time a cache file is accessed the id of that record will be added into the $fetched property of Doctrine_Cache
+ +2. At the end of each script the Doctrine_Cache destructor will write all these primary keys at the end of a stats.cache file + +3. Doctrine does propabalistic cache cleaning. The default interval is 200 page loads (= 200 constructed Doctrine_Managers). Basically this means +that the average number of page loads between cache cleans is 200. + +4. On every cache clean stats.cache files are being read and the least accessed cache files +(cache files that have the smallest id occurance in the stats file) are then deleted. +For example if the cache size is set to 200 and the number of files in cache is 300, then 100 least accessed files are being deleted. +Doctrine also clears every stats.cache file. + + |
"; + $c = file_get_contents("codes/$title - $t.php"); + $h->loadString($c); + print $h->toHtml(); + print " |
"; + $c = file_get_contents("codes/$name.php"); + $h->loadString($c); + print $h->toHtml(); + print " |
+ | + |
+
. + * + * @access public + * @param bool $return Return rather than print the results + * @param bool $linenum Display line numbers + * @param string $format Specify format of line numbers displayed + * @param bool $funcref Reference functions to the PHP manual + * @return string A HTML block of code + */ + function toHtml($return = false, $linenum = false, $format = null, $funcref = true) + { + // Ensure source has been loaded + if ($this->_source == false) { + return false; + } + + // Line numbering + if ($linenum === true && $format === null) { + $format = '%02d '; + } + + // Format code + $source = $this->toArray($funcref); + $out = " \n";
+ foreach ($source as $i => $line) {
+ $out .= ' ';
+
+ if ($linenum === true) {
+ $out .= sprintf($format, $i);
+ }
+
+ $out .= empty($line) ? ' ' : $line;
+ $out .= " \n";
+
+ if ($return === true) {
+ return $out;
+ } else {
+ echo $out;
+ }
+ }
+
+
+ /**
+ * Convert the source to formatted HTML blocks.
+ * Each line ends with . + * + * This method ensures only PHP is between blocks.
+ *
+ * @access public
+ * @param bool $return Return rather than print the results
+ * @param bool $linenum Display line numbers
+ * @param string $format Specify format of line numbers displayed
+ * @param bool $reset Reset the line numbering each block
+ * @param bool $funcref Reference functions to the PHP manual
+ * @return string A HTML block of code
+ */
+ function toHtmlBlocks($return = false, $linenum = false, $format = null, $reset = true, $funcref = true)
+ {
+ // Ensure source has been loaded
+ if ($this->_source == false) {
+ return false;
+ }
+
+ // Default line numbering
+ if ($linenum === true && $format === null) {
+ $format = '%03d ';
+ }
+
+ // Init
+ $source = $this->toArray($funcref, true);
+ $out = '';
+ $wasplain = true;
+ $k = 0;
+
+ // Loop through each line and decide which block to use
+ foreach ($source as $i => $line) {
+ // Empty line
+ if (empty($line)) {
+ if ($wasplain === true) {
+ $out .= ' ';
+ } else {
+ if (in_array($i+1, $this->_plaintextkeys)) {
+ $out .= " \n";
+
+ // Reset line numbers
+ if ($reset === true) {
+ $k = 0;
+ }
+ } else {
+ $out .= ' ';
+ // Add line number
+ if ($linenum === true) {
+ $out .= sprintf($format, ++$k);
+ }
+ }
+ }
+
+ // Plain text
+ } elseif (in_array($i, $this->_plaintextkeys)) {
+ if ($wasplain === false) {
+ $out .= "\n";
+
+ // Reset line numbers
+ if ($reset === true) {
+ $k = 0;
+ }
+ }
+
+ $wasplain = true;
+ $out .= str_replace(' ', ' ', $line);
+
+ // Code
+ } else {
+ if ($wasplain === true) {
+ $out .= "\n";
+ }
+ $wasplain = false;
+
+ $out .= ' ';
+ // Add line number
+ if ($linenum === true) {
+ $out .= sprintf($format, ++$k);
+ }
+ $out .= $line;
+ }
+
+ $out .= " \n";
+ }
+
+ // Output method
+ if ($return === true) {
+ return $out;
+ } else {
+ echo $out;
+ }
+ }
+
+
+ /**
+ * Assign a color based on the name of a token
+ *
+ * @access private
+ * @param int $token The token
+ * @return string The color of the token
+ */
+ function _token2color($token)
+ {
+
+ switch ($token):
+ case T_CONSTANT_ENCAPSED_STRING:
+ return $this->highlight['string'];
+ break;
+
+ case T_INLINE_HTML:
+ return $this->highlight['html'];
+ break;
+
+ case T_COMMENT:
+ case T_DOC_COMMENT:
+ case T_ML_COMMENT:
+ return $this->highlight['comment'];
+ break;
+
+ case T_ABSTRACT:
+ case T_ARRAY:
+ case T_ARRAY_CAST:
+ case T_AS:
+ case T_BOOLEAN_AND:
+ case T_BOOLEAN_OR:
+ case T_BOOL_CAST:
+ case T_BREAK:
+ case T_CASE:
+ case T_CATCH:
+ case T_CLASS:
+ case T_CLONE:
+ case T_CONCAT_EQUAL:
+ case T_CONTINUE:
+ case T_DEFAULT:
+ case T_DOUBLE_ARROW:
+ case T_DOUBLE_CAST:
+ case T_ECHO:
+ case T_ELSE:
+ case T_ELSEIF:
+ case T_EMPTY:
+ case T_ENDDECLARE:
+ case T_ENDFOR:
+ case T_ENDFOREACH:
+ case T_ENDIF:
+ case T_ENDSWITCH:
+ case T_ENDWHILE:
+ case T_END_HEREDOC:
+ case T_EXIT:
+ case T_EXTENDS:
+ case T_FINAL:
+ case T_FOREACH:
+ case T_FUNCTION:
+ case T_GLOBAL:
+ case T_IF:
+ case T_INC:
+ case T_INCLUDE:
+ case T_INCLUDE_ONCE:
+ case T_INSTANCEOF:
+ case T_INT_CAST:
+ case T_ISSET:
+ case T_IS_EQUAL:
+ case T_IS_IDENTICAL:
+ case T_IS_NOT_IDENTICAL:
+ case T_IS_SMALLER_OR_EQUAL:
+ case T_NEW:
+ case T_OBJECT_CAST:
+ case T_OBJECT_OPERATOR:
+ case T_PAAMAYIM_NEKUDOTAYIM:
+ case T_PRIVATE:
+ case T_PROTECTED:
+ case T_PUBLIC:
+ case T_REQUIRE:
+ case T_REQUIRE_ONCE:
+ case T_RETURN:
+ case T_SL:
+ case T_SL_EQUAL:
+ case T_SR:
+ case T_SR_EQUAL:
+ case T_START_HEREDOC:
+ case T_STATIC:
+ case T_STRING_CAST:
+ case T_THROW:
+ case T_TRY:
+ case T_UNSET_CAST:
+ case T_VAR:
+ case T_WHILE:
+ return $this->highlight['keyword'];
+ break;
+
+ case T_CLOSE_TAG:
+ case T_OPEN_TAG:
+ case T_OPEN_TAG_WITH_ECHO:
+ default:
+ return $this->highlight['default'];
+
+ endswitch;
+ }
+
+}
+
+?>
diff --git a/manual/images/docflash.jpg b/manual/images/docflash.jpg
new file mode 100644
index 000000000..0e0d6a71b
Binary files /dev/null and b/manual/images/docflash.jpg differ
diff --git a/manual/images/docflashbg.jpg b/manual/images/docflashbg.jpg
new file mode 100644
index 000000000..3821958db
Binary files /dev/null and b/manual/images/docflashbg.jpg differ
diff --git a/manual/images/docnew.gif b/manual/images/docnew.gif
new file mode 100644
index 000000000..60a9c7080
Binary files /dev/null and b/manual/images/docnew.gif differ
diff --git a/manual/images/gradient1.jpg b/manual/images/gradient1.jpg
new file mode 100644
index 000000000..254ead776
Binary files /dev/null and b/manual/images/gradient1.jpg differ
diff --git a/manual/images/gradient2.jpg b/manual/images/gradient2.jpg
new file mode 100644
index 000000000..4c94fd978
Binary files /dev/null and b/manual/images/gradient2.jpg differ
diff --git a/manual/images/gradient3.jpg b/manual/images/gradient3.jpg
new file mode 100644
index 000000000..cc86606f1
Binary files /dev/null and b/manual/images/gradient3.jpg differ
diff --git a/manual/images/logo.jpg b/manual/images/logo.jpg
new file mode 100644
index 000000000..8f3660ece
Binary files /dev/null and b/manual/images/logo.jpg differ
diff --git a/manual/images/tausta2.jpg b/manual/images/tausta2.jpg
new file mode 100644
index 000000000..b79d65692
Binary files /dev/null and b/manual/images/tausta2.jpg differ
diff --git a/manual/styles/basic.css b/manual/styles/basic.css
new file mode 100644
index 000000000..e2b640279
--- /dev/null
+++ b/manual/styles/basic.css
@@ -0,0 +1,49 @@
+body {
+margin: 10,0,0,0;
+font-family: Verdana, Arial, Helvetica;
+text-decoration: none;
+}
+
+hr {
+color: A50A3D;
+}
+hr.small {
+color: A50A3D;
+width: 100%;
+align: left;
+height: 1 px;
+}
+td.small {
+font-family: Verdana, Arial, Helvetica;
+font-size: 12 px;
+}
+dd {
+ line-height: 1.5em;
+}
+ul {
+ line-height: 1.5em;
+}
+td {
+font-size: 12 px;
+font-family: Verdana, Arial, Helvetica;
+}
+joo {
+color: A50A3D;
+}
+table.dashed {
+border-style: solid;
+border-color: 116699;
+border-width: 1 px;
+}
+b.title {
+color: #A50A3D;
+}
+u.title {
+color: #367FAC;
+}
+a.big {
+font-size: 13 px;
+}
+a {
+color: #367FAC;
+}
diff --git a/manual/styles/steel.css b/manual/styles/steel.css
new file mode 100644
index 000000000..87e9174f1
--- /dev/null
+++ b/manual/styles/steel.css
@@ -0,0 +1,45 @@
+body {
+margin: 10,0,0,0;
+font-family: Verdana, Arial, Helvetica;
+text-decoration: none;
+background: #393939;
+color: #dfdfdf;
+}
+hr {
+color: a3cb14;
+}
+hr.small {
+color: a3cb14;
+width: 100%;
+align: left;
+height: 1 px;
+}
+td.small {
+font-family: Verdana, Arial, Helvetica;
+font-size: 12 px;
+}
+td {
+font-size: 12 px;
+font-family: Verdana, Arial, Helvetica;
+background: #393939;
+}
+joo {
+color: A50A3D;
+}
+table.dashed {
+border-style: solid;
+border-color: 116699;
+border-width: 1 px;
+}
+b.title {
+color: #0090d9;
+}
+u.title {
+color: #367FAC;
+}
+a.big {
+font-size: 13 px;
+}
+a {
+color: #a3cb14;
+}
diff --git a/manual/top.php b/manual/top.php
new file mode 100644
index 000000000..7de1cb637
--- /dev/null
+++ b/manual/top.php
@@ -0,0 +1,22 @@
+
+
+
+
|