ongom / dite-orm
A PHP ORM, for interacting with the different kind of relational database.
Requires
- php: >=7
README
This is PHP ORM for interacting with relational databases. It supports only Sqlite, Mysql, Sqlserver and Postgresql databases. It allows us to keep oursevles within only PHP code instead of switching between sql and PHP code.
Features of dite-orm.
Dite-orm is too close to SQL. It can easily help to interact with the database quickly and you get everything you need in one line of code.
-
Supports multiple DBMS.
-
Configuration settings can be added directly into the .env file or using the setup method, but not both.
-
Uses PDO in the background.
-
Shorthand for the crud.
-
Provision for pagination, limit, order by, selecting only few fields, where clause, etc.
-
Geting record or records with their related data eg. getting posts with thier comments, or posts with the people who posted them.
-
Getting data from one table basing on another table. Example getting products that has no orders in the order table, getting products that has been ordered the most and those that has been oredered least in the order in which they have been ordered.
-
Relationship (one to many, one to one, many to many) both forward and backword relationship in just a single function.
-
Printing out the query that are running to produce the records given out.
-
Provision for writing raw SQL.
-
Schema for creating tables and their fields using the correct datatype customizeable to meet you needs.
-
Table name being queried matches the class name quering it. It can also allow for passing table name using DB::table().
-
And many more.
Sample usage
User::create(['user'=>'john doe']); //creating a user User::update(1,['user'=>'mike doe']); //updating a user with id 1 User::update(['age'=>20],['user'=>'mike doe']); //updating a user with id 20 User::delete(2); //deleting a user with age 20 User::delete(['age'=>20]); //deleting a user with age 20 User::all(['age'=>3]); //getting users with age 3 User::find()->get(); //getting all users DB::table('blog_post') ->limit(3) ->select('title, post_body') ->get();
Installation.
composer require ongom/dite-orm
Setup.
To get started, create a .env file at the root of your project, this is where you will add some configaration settings for dite-orm.
Setup for sqlite.
DRIVER = sqlite DATABASE_NAME = schooldb RUN_SCHEMA = 1
- DRIVER is the database management system you are using. The value is sqlite.
- DATABASE_NAME is the name of the database you are using. The value is the name of your database like 'school_database'.
Setup for mysql, sqlserver, postgre.
First create the database example 'schooldb' then add the following code to the .env.
DRIVER = mysql DATABASE_NAME = schooldb SERVER_NAME = localhost USER_NAME = root DATABASE_PASSWORD = 1234 RUN_SCHEMA = 1
-
DRIVER is the type of database management system you are using. List of posible values are;
- mysql
- sqlsever
- postgresql or pgsql
-
SERVER_NAME is the server name or port for example 'localhost' or 3605.
-
USER_NAME is the username e.g. example 'root'.
-
DATABASE_PASSWORD is the password of the database example '23R42'.
Other configarations that can be added to .env file.
IS_DEVMODE = 1 FETCH_MODE = std_array RUN_SCHEMA = 1 SQL_COLOR = green
- IS_DEVMODE is for debuging purpose.
- 1 means you want to print queries and messeges on the screen, this should be only used in development.
- 0 or any other number means no printing queries or messeges on the screen, it is used for production.
- RUN_SCHEMA. When you try to create the table using the Schema, the table will not be created, to solve this add RUN_SCHEMA = 1 to the .env file. After creating the tables you can turn the value 0 (RUN_SCHEMA = 0) or remove it completely to avoid rerunning the queries for creating the table again. When you add any new tables, you will have to turn it back to 1 and again run the code.
- FETCH_MODE defines how the records are going to be fetched from the database.
- std_arrays means that the records will be fetched as a php standard array whose values can be access as shown bellow.
$user = ["name"=>"tom", "age"=>36]; $user["name"]; //tom $user["age"]; //36
- std_obj means records will be fetched as stdClass object which can be access using arrow syntax as below.
$user = stdClass Object ([name] => tom [email] => tom@gmail.com) $user->name; //tom $user->age; //36
- SQL_COLOR definds the color of sql keywords when printing to the screen. It has a default purple color.
- NONSQL_COLOR definds the color of non sql keywords when printing to the screen. It has a default whitish color.
- SQL_BG definds the background color of the sql being printed on screen in dev mode. The default is black.
- FULL_SQL It can allow you to see the prepared sql and the value with which it will be executed. 0 to turn it on or 1 to turn it off. If it is not added , it defaults to full sql being written.
NOTE: All colour codes are defind in the .env file as :fff but not #fff (Use :
but not #
)
Alternative way of creating connection to the database Incase you don't want to add all the connetion details directly to the .env file, use the static method ,setup() which is on both Model and Schema from dite. This method take in an array of configuration setup
Model::setup([ 'DRIVER' => 'sqlite', 'DATABASE_NAME' => 'schooldb', 'RUN_SCHEMA' => 1, 'APP_PASSWORD' => 'rqae hrue bili alru', 'LOGGER' => 1, 'SQL_COLOR' => ':3f2', 'NONSQL_COLOR' = ':f3fb', 'SQL_BG' = 'red', 'SEPARATOR' = 'white' ]); // And also Schema::setup([ 'DRIVER' => 'sqlite', 'DATABASE_NAME' => 'schooldb', 'RUN_SCHEMA' => 1, 'APP_PASSWORD' => 'rqae hrue bili alru', 'IS_DEVMODE' => 1, 'SQL_COLOR' => ':3f2', 'NONSQL_COLOR' => ':f3fb', 'SQL_BG' => 'red', 'SEPARATOR' =>'white' ]);
Creating tables.
- Using Dite Schema.
- Using existing database / Using other softwares (mysql workbench, navycat, DB browser, etc. ) to create.
1. Using existing database or other softwares to create the database.
When using other software like myql workbench or PHPmyadmin, all you need to care about is the primary key field. Primary key field name is got from the name of the table written in lowercase. ie
- Users pk fieldl be users_id
- BlogPost pk fieldl be blog_post_id
- Blog_post pk fieldl be blog_post_id
- Prices pk fieldl be Prices_id
Note:: Uppercase letter in the middle of the name result into underscore before the uppercase letter.
Creating connetion to db.
Add connection to the database direct to the .env or use the setup method to add connection. Then the model classes(representing each table) are defined like below.
use Dite\Model\Model; require_once "path/to/vendor/autoload.php"; class DB extends Model{ public function __constructor(){ self::setup([ 'DRIVER' => 'sqlite', 'DATABASE_NAME' => 'schooldb', 'RUN_SCHEMA' => 1, 'LOGGER' => 1, 'SQL_COLOR' => ':3f2', ]); } } //users class Users extends DB{} //posts class Posts extends DB{} // Status class Status extends DB{}
2. Using Dite Schema to create the database.
Add the following code below the require statement.
use Dite\Schema\Schema; use Dite\Model\Model; use Dite\Table\Table; require_once "path/to/vendor/autoload.php"; //users class Users extends Model{ public function __construct() { Schema::create(Users::class, function(Table $table){ $table->id(); $table->string('name'); $table->enum('gender', ['male','female']); $table->string('email')->unique()->notnull(); $table->int('age')->notnull(); $table->timestamp(); }); } }
The above code defines a table called users with the following fields.
- users_id - integer type, primary key, autoincrement and not null.
- name - varchar(255).
- gender - enum that accept only male or female as values.
- email - unique and not null.
- age - int and not null.
- created_at - default current timestamp.
- updated_at - default current timestamp that updateds when the record updates.
You can go ahead and add a post table. Below is the overall code on how to create three tables (users, posts, status) including foreign keys.
//users class Users extends Model{ public function __construct() { Schema::create(Users::class, function(Table $table){ $table->id(); $table->string('name'); $table->string('email')->unique()->notnull(); $table->int('age')->notnull(); $table->timestamp(); }); } } //posts class BlogPosts extends Model{ public function __construct() { Schema::create('BlogPosts', function(Table $table){ $table->id(); $table->string('title')->notnull(); $table->string('body')->notnull(); $table->foreignkey('users_id')->notnull(); $table->foreignkey('status_id')->cascade(); $table->timestamp(); }); } } // Status class Status extends Model{ public function __construct() { Schema::create(Status::class, function(Table $table){ $table->id(); $table->int('status')->notnull(); }); } } //Instantiating the three classes $status = new Status() $users = new Users() $posts = new BlogPosts()
Note:
- We have instantiated the classes starting with Status.This is because the status table is being referenced by Posts in the foreign key fielded and the table have to be created first before its is referenced.Tables that are being reference must be above. If you change the order like below, you will get a foriegn key error mostly in mysql.
//Instantiating the three classes $users = new Users() $posts = new Posts() $status = new Status()
Run the code e.g. by opening your file in the browser. This will create the table in the database.
- You can also use Users::class or 'Users' for table name called users and so for other tables.
After the tables has bean created, You can open .env and change RUN_SCHEMA = 0 or else the tables will try to be recreated.
1. Creating an intermidate table.
The intemediate table has a convention of creating it inorder for Dite to understand. You have to concatenate the two table names. For example teachers and courses tables, the intermediate table will be teachers_courses or TeachersCourses::class and the primary key fieldl be teachers_courses_id . The intermediate table must be created like below.
//Teachers table class Teachers extends Model{ public function __construct() { Schema::create(Teachers::class, function(Table $table){ $table->id(); $table->string('teacher_name'); $table->string('email'); }); } } //Courses table class Courses extends Model{ public function __construct() { Schema::create(Courses::class, function(Table $table){ $table->id(); $table->string('course_name'); }); } } //Intermediate table class TeachersCourses extends Model{ public function __construct() { Schema::create(TeachersCourses::class, function(Table $table){ $table->id(); $table->foriegnKey('courses_id'); $table->foriegnKey('teachers_id'); }); } }
Meaning of each of the methods used for building the table.
- id() - Defines an autoincrementing primary id field set it not null. You can not chain any method on to id().
$table->id();
- primarykey() - Defines primay key field
$table->primaryKey();
- string() -Sql varchar field takes in two parameter, one mandatory string parameter(field name like comments), second optional integers parameter which defaults to 255 (max length of the charactors accepted).
$table->string('name'); //OR $table->string('name', 50);
- text() - Sql text field. It takes in two parameter, one mandatory string parameter(field name like comments), second optional integers parameter which defaults to 65535 (max length of the charactors accepted).
$table->text('name'); //OR $table->text('name', 200);
- longText() - Sql text field. It takes in one mandatory string parameter(field name like Posts).
$table->longText('name');
- int() - Sql integer field. It takes in one mandatory string parameter(field name like comments).
$table->int('size');
- bigint() -Sql big interger field. It takes in one mandatory string parameter(field name like comments).
$table->bigInt('size'); //OR $table->bigInt('size', 6);
- enum() - It takes in two mandatory parameter, one string parameter(field name like size), second array parameter(only values accepted).
$table->enum('size', ['small','medium', 'large']);
- unsigned() - Sql unsign datatype field. It take in field name
$table->unsign('year');
- boolean() - Sql boolean datatype field(true or false). It take in field name.
$table->boolean('is_active');
- float() - Sql floating point datatype field. It take in field name
$table->float('distance');
- double() -Sql double datatype field. It take field in name.
$table->double('distances');
- decimal() -Sql decimal datatype field. It take in field name. This may not be avilable in other DRIVER
$table->decimal('distance');
- year() -Sql year datatype field. It take in field name.
$table->year('Year_of_birth');
- timestamp() -Sql timestamp. It will create the created_at and updated_at columns.
$table->timestamp();
- sql() -This takes in a string parameter, the query for creating a table. You must not chain anything on to this method
$table->sql("CREATE TABLE IF NOT EXISTS Users ( status_id INT AUTO_INCREMENT PRIMARY KEY NOT NULL , status INT NOT NULL , created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP )");
- primarykey() -Sql primarykey key field..
$table->string('user_id')->primarykey(); // OR $table->int('user_id')->primarykey(); // OR $table->int('user_id')->primarykey()->autoincrement();
- foreignkey() -Sql foreign key field. It take in the name of the fieldng referenced and an optional type which default to
INT
. Use the same type as used in the primary key field egINT
,VARCHAR
,TEXT
orLONGTEXT
. You only need to suply the type if you are not using typeINT
or$table->id()
for primary key.$table->foreignkey('user_id' 'INT');
Field constrains
-
notnull() - Sql NOT NULL constrain. You can not do this on id() method..
$table->foreignkey('user_id')->notnull(); $table->string('user_name')->notnull(); $table->id('user_name')->notnull(); // dont do this!!
-
unique() -Sql UNIQUE constrain.
$table->foreignkey('posts-id')->unique(); $table->email('user_email')->unique()->notnull();
-
cascade() -This is chained only on foriegn ky field it is optional. Its sets ON DELETE and ON UPDATE constrian to CASCADE.
$table->foreignkey('post_id')->cascades(); $table->foreignkey('user_id')->unique()->notnull()->cascade();
-
cascade() -This is chained only on foriegn key field it is optional. It sets ON DELETE and ON UPDATE constrian to CASCADE.
$table->foreignkey('post_id')->cascades(); $table->foreignkey('user_id')->unique()->notnull()->cascade();
-
restrict() - It sets ON DELETE and ON UPDATE constrian to RESTRICT.
$table->foreignkey('post_id')->restrict();
-
setnull() - It sets ON DELETE and ON UPDATE constrian to SET NULL.
$table->foreignkey('post_id')->setnull();
-
noaction() - It sets ON DELETE and ON UPDATE constrian to NO ACTION.
$table->foreignkey('post_id')->noaction();
You can also set this costrain one by one as shown below
-
cascadeDelete() - It sets ON DELETE CASCADE.
$table->foreignkey('post_id')->cascadeDelete();
-
cascadeUpdate() - It sets ON UPDATE CASCADE.
$table->foreignkey('post_id')->cascadeUpdate();
-
restrictDelete() - It sets ON DELETE RESTRICT.
$table->foreignkey('post_id')->restrictDelete();
-
restrictUpdate() - It sets ON UPDATE RESTRICT.
$table->foreignkey('post_id')->restrictUpdate();
-
setnullDelete() - It sets ON DELETE SET NULL.
$table->foreignkey('post_id')->setnullDelete();
-
setnullUpdate() - It sets ON UPDATE SET NULL.
$table->foreignkey('post_id')->setnullUpdate();
-
noactionDelete() - It sets ON DELETE NOACTION.
$table->foreignkey('post_id')->noactionDelete();
-
noactionUpdate() - It sets ON UPDATE NOACTION.
$table->foreignkey('post_id')->noactionUpdate();
Below is a very valid chain.
$table->foreignkey('post_id')->noactionUpdate()->setnullDelete();
Querying the database.
This section will teach us how to create, read, update and delete record.
Create new record.
class BlogPost queries the tabl blog_post and ComponyActiveEmployees queries compony_active_employees
If you are using autoincreamenting id from $table->id()
, don't pass the primary key fieldly pass primary key fieldprimary key is from $table->string('user_id')->primarykey()
or $table->int('user_id')->unique()->notnull()
.
// creating single user $user = User::create(["user_name"=>"tom", "age"=>32]); //creates multiple users at once $user = User::create([ ["user_name"=>"tom", "age"=>32], ["user_name"=>"mike", "age"=>25], ["user_name"=>"loy", "age"=>27], ]);
Don't pass created_at and updated_at fieldause they get fieldautomatically.
The create method returns the new record created only if you are creating a single record.
Updating record.
The static method update() is used, is takes in two paramters, the record you want to update (id or where clause) the and associative array of the new values
//updating the user with id 1 $user = User::update(1, ["user_name"=>"tom", "age"=>32]); //updating the user where age > 10 $user = User::update(["age"=>[":gt"=>10]], ["user_name"=>"john", "age"=>32]);
The update method returns the new updated record ony if you have passed the id.
Deleting record.
Static method delete will delete a record by it's id.
//deleting the user where user_id = 1 $user = User::delete(1); //deleting the user where user_name = tom $user = User::delete(["user_name"=>"tom"]); //deleting the user where IN (1,2) $user = User::delete([1,2]);
NB:
Part of the query that makes the where clause is passed as a parameter to the method
Counting the number of records that matcht the query.
Static method countRows() is used for Counting the number of records that match the query.
//counting number the users all users $user = User::countRows(); //updating the user where user_name = tom and age = 30 $user = User::countRows(["user_name"=>"tom", "age" => 30]);
Reading records from the database.
This can be done using many methods which are all discussed below. All static methods that are used for reading records without chaining functionality takes in atleast two optional parameters, the where clause array / int and the fieldu want back. The fields you want back can be passed as comma separated string value of the columns you want back OR as an array of all the fieldu want back. As shown below
//get all the records from the table user $user = User::all(null, 'name, email'); //get all the records from the user table where age > 30 $user = User::all(["age"=>[":gt"=>30]], ['name', 'email']);
This is a static method that gets all the records that matches the query.
//get all the records from the table user $user = User::all(); ////get all the records from the user table where age > 30 $user = User::all(["age"=>[":gt"=>30]]);
//get all the records from the table user $user = User::first(); //get the first or first 5 record limited from the user table where age > 30 $user = User::first(["age"=>[":gt"=>30]], 'name, age',5);
//get the last records from the table user $user = User::last(); //get the last records from the user table where age > 30 $user = User::last(["age"=>[":gt"=>30]]);
//get the records from the table user where user_id = 2 $user = User::findById(2); //get the records from the table user where user_id IN (1,2,3) $user = User::findById([1,2,3]);
This is a method that gets records that matches the query.
//get the one and first record from the table user where user_id > 10 $user = User::findOne(["user_id"=>[":gt"=>10]]);
This is a method that checks if the record exists. It returns a true or false.
//checks if any user record exist $user = User::exist(); //checks if user with user_id 2 exists $user = User::exist(2); //checks if the users with user_id greater than 10 exist $user = User::exist(["user_id"=>[":gt"=>10]]); //checks if email tom@gmail.com exist in user table $user = User::exist(["email"=>"tom@gmail.com"]);
-
sql().
This is a static method inherited from Model class. It takes in two parameter, An sql query and the values you want to bind to the query. This can be usefull when writing a much complext query.
$user = User::sql("SELECT * FROM users"); $user = User::sql("SELECT * FROM users", []); $user = User::sql("SELECT * FROM users WHERE user_name = ? And age > ?", ['tom', 30]);
joins
There are several ways of joining tables. All joins method takes in three parameter, table name, an optional join condition, optional where clause and optional select. Dite will try set its own join conditon as below if the condition is not pased. "first_table.first_table_id = second_table.first_table_id"
$user = User::joins(Post::class, "user.user_id = post.user_id", $orderby = "name ASC|DESC", $select = 'user.name, post.title');
This two methods do the same thing , they will inner join the two the tables.
$user = User::joins(Post::class); //OR $user = User::joins("Post"); //select * from user join post on user.user_id = post.user_id
There are methods for other type of joins.
- fullJoins()
- leftJoins()
- rightJoins()
- innerJoins()
- rightOuterJoins()
- leftOuterJoins()
So far we have looked at only static methods without chaining, Let us look at other chaining functionality available.
find().
For getting array or one of the records from a table.
$users = User::find()->get(); //select * from user $users = User::find(5)->get(); //select * from user where user_id = 5
There are few more methods you can chain onto the method find() as described below.
orderBy().
Will sort the result in descending or asscending order .
$users = User::find()->orderBy("user_id, name ASC, created_at DESC")->get(); //select * from user order by user_id asc name asc created_at DESC. $user = User::find()->orderBy(['user_id'=> 'asc'])->get(); //select * from user order by user_id desc. $user = User::find()->orderBy(['user_id'=> 'asc', 'name'=>'desc'])->get(); //select * from user order by user_id desc age asc.
groupBy()
This will group the result by the suplied fieldoupBy() takes in a string parameter.
$user = User::find()->groupBy('name')->get(); //select * from user group by name. $user = User::find()->groupBy('group_id')->get(); //select * from user group by group_id. $user = User::find()->groupBy('username')->get(); //select * from user group by username where name = john doe.
select().
This will select only the spacified fieldhe parameter is either string or arrays.
$users = User::find()->select(['name', 'age'])->get(); //select name, age from user. //OR $users = User::find()->select('name AS staff_names, age')->get(); //select name, age from user. $users = User::find() ->select(['name as names_of_staffs', 'age', "COUNT(name) * 2 as total"]) ->get();
limit().
This is used to spacify the number of records you want to fetch. It defaults to 12.
$users = User::find()->limit(5)->get(); //select * from user limit 5.
skip() or offset().
Both of these do the same thing. They are used to spacify the number of records that will be skiped. It defaults to 0.
$users = User::find()->skip(10)->get(); //OR $users = User::find()->offset(10)->get();
A combination of skip and limit can be used for paginating your result.
$users = Post::find()->skip(5)->limit(10)->get(); //select * from user limit 5,10
Pagination.
Pagination helps to query only a slice of records from the database. It has two methods;
- page() - Takes in the page you want.
- perpage() - Takes number of records for each page.
$users = User::find() ->page(2) ->perpage(10) ->get();
- page() and perpage() methods will activate pagination
The above query will return something like below
[ 'num_records' => 15; //total number of records 'num_pages' => 2; //total number of pages 'has_next' => false; //if it has next page or not 'current_page' => 2; //the current page showing 'has_prev' => true; //if it has previous page or not 'next_page' => null; //what next page is, null for no next page 'prev_page' => 1; //what next page is, null for no next page 'per_page' => 10; //number of records per page 'position' => 1; //position of the first record of a page in the entire result 'data' => [...]; // records for each page ]
Using Model class to query data.
Pass the name of the table to the model constrctor if you want to query using the Model class.
class DB extends Model{} $Post = DB::table('Post')->get(); // OR $Posts = Model::table('Post')->get(); // $users = DB::table('user')->select('name')->get(); // find $users = Model::table('user') ->limit(10) ->offset(5) ->select('name') ->get(); // paginating $users = Model::table('post') ->page(10) ->perpage(5) ->select('title') ->get(); // OR $users = DB::table('post') ->page(10) ->perpage(5) ->select('title') ->get();
You can chain any valid method like select, join, group, etc.
Also
$user = DB::table('user')::all(); $user = DB::table('user')::findById(4);
Other chaining method.
There are other chaining methods.
- withAll()
It takes in three parameters; table name , optional where clause and selected field It will add all the results from the second table keeping the id of the record in the first table. E.g. It retrieve the user or users with all their posts. Works in a one to one and one to many relationship.
$user = User::find() ->withAll(Post::class, ['status'=>'active'],'title', 'title DESC') ->get(); // $user = User::find(5) ->withAll(Post::class, ['status'=>'active'], 'title', 'title DESC') ->get(); //You can chain it many time like $user = Product::find(5) ->withAll(Order::class) ->withAll(Status::class) ->get();
- withOne()
It takes in three parameters; table name , optional where clause and selected field It will add one result from the second table keeping the id of the record in the first table. E.g. It retrieve the user or users with one of thier posts. Works in a one to one and one to many relationship.
$user = User::find() ->withOne(Post::class, ['status'=>'active'],'title') ->get(); //Returns users post where you can paginate, each having there post appended to $user = User::find(3) ->withOne(Post::class, ['status'=>'active'],'title') ->get(); //Returns on post by its id together with one of his post //You can chain it many time like $user = Product::find(5) ->withOne(Order::class) ->withOne(Status::class) ->withOne(Owner::class) ->page(1) ->get();
- attach()
It takes in three parameters; table name, optional where clause and selected field It will add all the results from the second second whose id is in the first table. E.g. It retrieve the post or posts with person who posted it. Works in a one to one relationship.
$user = Post::find() ->attach(User::class) ->get(); //Returns posts together with the user wo posted it. $user = Post::find() ->attach(User::class) ->attach(Department::class) ->get(); //Returns a post together with the user who posted it. //You can chain it many time like $user = User::find(5) ->attach(Department::class) ->withOne(City::class) ->get();
- withMost()
It gets the records from first table whose id has appeared the most in the second table in a one to many relationship. It takes in one parameter; table name. Chain the limit() to limit tha number of result.
$user = User::withMost(Post::class) ->limit(5) ->get(); //returns 5 users that has posted the most. // $user = User::withMost(Post::class) ->withAll(Post::class) ->attach(City::class) ->limit(5) ->get();
- withLeast()
It gets the records from first table whose id has appeared least in the second table in a one to many relationship. It takes in one parameter; table name. Chain the limit() to limit tha number of result.
$user = User::withLeast(Post::class) ->withAll(Post::class) ->attach(City::class) ->limit(5) ->get(); //Returns 5 users that has least number of post.
- withOut()
It gets the records from first table whose id has not appeared in the second table in a one to many relationships. It takes in one parameter; table name. Chain the limit() to limit tha number of result. You can also limit, paginate, etc.
$user = User::withOut(Post::class) ->get(); //Returns users that has no post.
withThrough()
It takes in three parameters; table name,where and select. It retrieves teaches with the subjects they teach. This happens in a many to many relationship.
$user = Teacher::find() ->withThrough(Course::class,['status'=>1], 'coures_name, course_id') ->get();
- attachThrough()
It takes in three parameters; table name,where and select. It will add all the Teachers teaching a perticuler course. This happens in a many to many relationship.
$user = Course::find() ->attachThrough(Teacher::class,['status'=>1], 'name, age') ->get(); //Returns Courses together with the correspnding teachers teaching each.
Joining tables.
join().
Earlier we looked at joins but we were able to join only two tables using the static methods, now lets join more than two tabes. These are the different types of joins which are avaiable;
- fullJoin()
- join()
- innerJoin()
- leftJoin()
- rightJoin()
- letftOuterJoin()
- rightOuterJoin()
- leftOuterJoin()
$users = User::find() ->join('post') ->join('comments', 'user.user_id = comment.user_id') ->get();
Three tables will be involved in this join, user, post and comments. You can join over 20 differnt tables together using any of the above types of join and apply pagination, select order by, etc like below.
Remember all joins takes in one more optional parameter i.e. the the join condition.
$users = User::find() ->join('post') ->leftJoin('comments') ->page(1) ->perpage(10) ->orderBy('user.name'=>'desc') ->select(['user.name', 'post.title', 'count(*) As total']) ->where(["first_name" = "mike"]) ->get();
Where clause.
The where clause is passed as a parameter in the following methods.
- ::all()
- ::findOne()
- ::findbyId()
- ::first()
- ::last()
- ::delete()
- ::deleteMany()
- ::countRows()
- ::exist()
- ::find()
You can also chain the where() method on the following methods
- ::find()
- ::table()
It can be passed in the following ways.
1. Passing an integer.
When you pass an integer to methods like findById() or delete(), the integer is primary id of the record you will get back.
$result = User::findById(2); //OR $post = Post::find(2)->get();
The above code will return a single record whose primary id is 2.
2. Passing an associative array.
When you pass an associative array to methods like findOne() or all(), the array generate the query as bellow
$result = User::all(["user_id"=>5]); //SELECT * FROM user WHERE user_id = 5 //OR $result = User::all(["user_id"=>5, "name"=>"tom"]); //SELECT * FROM user WHERE user_id = 5 AND name = tom //OR $result = Posts::find() ->where(["user_id"=>5, "name"=>"tom"]) ->get(); //SELECT * FROM user WHERE user_id = 5 AND name = tom
3. Passing nested associative array.
Sometimes you want to apply operators like <, >, <=, >=, =, like, etc. this is done the following ways
$result = User::all(["user_id"=>['$lt'=>5]]); //SELECT * FROM user WHERE user_id < 5 //OR $result = User::all([ "user_id"=>[':lt'=>5], "age"=>[':gt'=>30], ]); //OR $result = User::all([ "email"=>"tom@gmail.com", "user_id"=>[':lt'=>5], "age"=>[':gt'=>30], ]); //SELECT * FROM users WHERE email = tom@gmail.com AND user_id < 5 AND age > 30
Instead of using ':', you can use '$', for example, the output of this code is the same.
$result = User::all(["user_id"=>['$lt'=>5]]); $result = User::all(["user_id"=>[':lt'=>5]]); //Both will output //SELECT * FROM user WHERE user_id < 5
4. Passing associative array where key is $and or :and.
This will only write queries in which the where clause is separated by AND.
$result = User::all([ ':and'=>[ "user_id"=>['$lt'=>5], 'age'=>[':gt'=>30], 'phone'=>3 333 333 333 ] ]); //SELECT * FROM user WHERE user_id < 5 AND age >30 AND phone = 3 333 333 333
5. Passing associative array where key is $or or :or.
This will only write queries in which the where clause is separated by OR.
$result = User::all([ ':or'=>[ "user_id"=>['$lt'=>5], 'age'=>[':gt'=>30], 'phone'=>3 333 333 333 ] ]); //SELECT * FROM user WHERE user_id < 5 OR age > 30 OR phone = 3 333 333 333 $result = User::all([ ':or'=>[ ['name'=>'tom', 'email'=>'tom@gmai.com'], "user_id"=>['$lt'=>5], 'age'=>[':gt'=>30], 'phone'=>3 333 333 333 ] ]); /* SELECT * FROM user WHERE (name = tom AND email = tom@gmai.com) OR user_id < 5 OR age > 30 OR phone = 3 333 333 333 */
6. Passing associative array where key is $nand or :nand.
This will negate the entire :and.
$result = User::all([ ':nand'=>[ "user_id"=>['$lt'=>5], 'age'=>[':gt'=>30], 'phone'=>3 333 333 333 ] ]); /* SELECT * FROM user WHERE NOT ( user_id < 5 AND age > 30 AND phone = 3 333 333 333) */ $result = User::all([ ':nand'=>[ "name" => "tom", "user_id"=>['$lt'=>5], ] ]); /* SELECT * FROM user WHERE NOT (name = tom AND user_id < 5 ) */
7. Passing associative array where key is $nor or :nor.
This will negate the entire :nor.
$result = User::all([ ':nor'=>[ "user_id"=>['$lt'=>5], 'age'=>[':gt'=>30], 'phone'=>3 333 333 333 ] ]); /* SELECT * FROM user WHERE NOT ( OR user_id < 5 OR age > 30 OR phone = 3 333 333 333) */ $result = User::all([ ':nor'=>[ "name" => "tom", "user_id"=>['$lt'=>5], ] ]); /* SELECT * FROM user WHERE NOT (name = tom OR user_id < 5 ) */
The same way, putting n just after $ or : in the operator will negate that part of the query,
$result = Users::all(["name" => "tom","user_id"=>['$nlt'=>5]]); /* SELECT * FROM users WHERE name = tom AND NOT (user_id < 5 )) */
Passing the same column name more than one times.
If the same key is going to appear more than once, normally associative array will only pick up the key which is written last, to deal with this kind of behavior , you have to append a leading underscore (_) before the column name in the array key, this is demonstrated below.
$result = Users::all(["name" => "joyce","_name"=>"tom"]) // SELECT * FROM users WHERE name = tom AND name = tom )) $result = Users::all([':or'=>[ "name" => "tom", "_name"=>"daniel", "__name"=>"loy"]]); // SELECT * FROM users WHERE name = tom OR name = deniel OR name = loy)) //Alternatively $result = Users::find() ->where([':or'=>[ "name" => "tom", "_name"=>"daniel", "__name"=>"loy"]]) ->get(); // SELECT * FROM users WHERE name = tom OR name = deniel OR name = loy))
You will have to append many underscores if the column name is repeating many times in that same associative array.
List of available operators for the where clause.
Operators | Sparcles Symbol | Example |
---|---|---|
= | $eq or := | ['name'=>[':eq' => 'tom']] OR ['name'=>[':=' => 'tom']] |
< | $< or :lt | ['age'=>[':lt' => 18]] |
> | $> or :gt | ['age'=>[':gt' => 18]] |
>= | $>= or :gte | ['price'=>[':gte' => 1000]] |
<= | $<= or :lte | ['price'=>[':eq' => 50]] |
Like | $like | ['name'=>[':like'=>'%micheal']] or ['name'=>[':like'=>'%cheal%']] |
Regexp | :regexp | ['name'=>[':regexp'=>'^mich']] or ['name'=>[':regexp'=>'cheal$']] |
In | :in | ['name'=>[':in' => ['tom','mike', 'joy']]] |
Between | :btn or :between | ['age'=>[':btn' => [20,30]]] |
Null | null | ['age'=>'null'] |
Not Null | not null | ['age'=>'not null'] |
Not < | $n< or :nlt | ['age'=>[':nlt' => 18]] |
Not > | $n> or :ngt | ['age'=>[':ngt' => 18]] |
Not >= | $n>= or :ngte | ['price'=>[':ngte' => 1000]] |
Not <= | $n<= or :nlte | ['price'=>[':neq' => 50]] |
Not Like | :nlike | ['name'=>[':nlike'=>'%micheal']] or ['name'=>[':nlike'=>'%cheal%']] |
Not In | :nin | ['name'=>[':nin' => ['tom','mike', 'joy']]] |
Not Between | :nbtn or :nbetween | ['age'=>[':nbtn' => [20,30]]] |
And | $and | [':and'=>[ 'age'=>10, 'name' =>tom ]] |
Not And | $nand | [':nand'=>[ 'age'=>10, 'name' =>tom ]] |
Or | $or | [':or'=>[ 'age'=>10, '_age' =>20 ]] |
Not or | $nor | [':nor'=>[ 'age'=>10, '_age' =>20 ]] |
Not Regexp | $nregexp | ['name'=>[':nregexp'=>'^mich']] or ['name'=>[':nregexp'=>'cheal$']] |
Relationships.
This will establish connection between some tables, for example if you have a post, you can easily get its comments, or if you have a user you can get all his posts. There are three types of relationships you can use here,
- One To One relationship.
- One To Many relationship.
- Many To Many relationship.
1. One To One relationship.
One user has one credit card and a credit card belongs to one user. To establish a One To One relationship here , you have to create one function in the Users class and CreditCards calss defination as shonw below.
$card = User::find(4) ->hasOne(CreditCard::class) ->get(); // returns one creditcard or false //and also $card = CreditCards::find(4) ->belongsToOne(User::class) ->get(); // returns one user or false
2. One To Many relationship.
The code will be as below.
$card = User::find(4) ->hasMany(Post::class) ->get(); // returns array of post records // and also $Post = Post::find(4) ->belongsToOne(User::class) ->get(); // returns a user or false
3. Many To Many relationship.
This will only work if you had created an intermediate table for the two tables. The intemediate table has a convention of creating it inorder for Dite to understand. You have to concatenate the two table names. For example teachers and courses tables, the intermediate table will be teachers_courses and the primary key fieldl be teachers_courses_id .The intermediate tabble must be created like below.
//Teachers table class Teachers extends Model{ public function __construct() { Schema::create(Teachers::class, function(Table $table){ $table->id(); $table->string('teacher_name'); $table->string('email'); }); } } //Courses table class Courses extends Model{ public function __construct() { Schema::create(Courses::class, function(Table $table){ $table->id(); $table->string('course_name'); }); } } //Intermediate table class TeachersCourses extends Model{ public function __construct() { Schema::create(TeachersCourses::class, function(Table $table){ $table->id(); $table->foriegnKey('courses_id'); $table->foriegnKey('teachers_id'); }); } }
Each time you create a teacher or a course remember to update the intermediate table.
Let's define the relationship.
$card = Teacher::find(4) ->hasManyMany(Courses::class) ->get(); // returns one user or false
NB On to any relationship, you can chain any valid method chain exept ->find()
$post = Users::find(2) ->hasMany(Post::class) ->select('post_id,title, post') ->orderBy('age') ->where(['title'=>['$like'=>'%computer']]) ->limit(10) ->get();
Dropping database table
This is done using the static method drop() like below. It returns a boolean , true for successful deleting and false for failure to delete the table.
Users::drop();