Create Laravel Api with samark/lumpineevill package | สร้าง Laravel API อย่างง่ายด้วย samark/lumpineevill package


ต่อยอดจากบทความที่แล้ว คือการสร้าง package ส่วนตัวด้วย composer แล้ว submit ไปที่ packagist.org ซึงตอนนี้ผมก็เลยสร้าง package ที่ใช้ทำมาหากินของผมในส่วนงานที่เกี่ยวข้องโดยตรงกับการทำ API ผมพบว่า API มันมี patten เนื้องานซ้ำไปซ้ำมา โดยหน้าที่หลัก ๆ ของมันมี 4 อย่างด้วยกันคือ

  • อ่านข้อมูล (Read)
  • เขียนข้อมูล ( Create)
  • แก้ไขข้อมูล (Update)
  • ลบข้อมูล ( Delete)
หรือที่โปรแกรมเมอร์ชอบเรียกรวม ๆ กันว่า ( CRUD ) นั่นเอง สำหรับ package ที่สร้างขึ้นมามีวัตถุประสงค์ดังนี้
  • ลดการทำงานของ Programmer 
  • สร้างไฟล์ที่สามารถใช้งานซ้ำได้
  • สร้างไฟล์ส่วนที่มีความจำเป็นให้อัตโนมัติ
  • เชื่อมโยงไฟล์เข้าด้วยกันแบบอัตโนมัติ
  • เพื่อผลิตงานให้ได้ไวที่สุด

ลดการทำงานของ Programmer
ผมพบว่าการสร้าง API หนึ่งเส้นจะต้องมีไฟล์ที่เกี่ยวข้องแบบน้อยสุด 4 ไฟล์ คือ
  1. route 
  2. controller
  3. model 
  4. migration (กรณีที่ยังไม่ได้ทำ migration เอาไว้)
จะเห็นว่านี่เป็น API patten ที่แบบทั่วไปซึ่งมันก็สะท้อนว่าถ้าหากเราจะพัฒนาไปใช้ patten อื่น ๆมันต้องมีไฟล์ที่เข้ามาเกี่ยวข้องเยอะกว่านี้แน่นอนเพื่อให้มันทำงานออกมาได้ดีและลดความซับซ้อนของ code ให้มากที่สุด ตัวอย่าง patten ที่ผมใช้คือ Repository patten จะมีไฟล์ที่เข้ามาเกี่ยวข้องดังนี้
  1. route
  2. middleware
  3. request
  4. controller
  5. repository
  6. model
  7. migration (กรณีที่ยังไม่ได้ทำ migration เอาไว้)
ภาพรวมของ Repository patten มันจะทำงานตามลำดับที่ผมเรียงไว้ด้านบน อธิบายคร่าวการทำงานของมันจะเกิดขึ้นเมื่อ Client create http request เข้ามาผ่าน route เช่น 
  • url : {base_url}/api/v1/contents
  • method : get 
  • authorization : JWT  [การทำ authentication รูปแบบหนึ่ง]
จากนั้นก่อนที่จะส่งข้อมูลไปทำงานจะถูก middleware กรองก่อนชั้นแรก เช่น api ที่เรียกเข้ามาผ่านการ authorization หรือเปล่า หลังจากนั้นจะถูกส่งต่อไปที่ requst ตัวอย่างไฟล์ request มีดังนี้
  • <?php
    /**
    * @author samark chaisanguan
    * @email samarkchsngn@gmail.com
    */
    namespace App\Http\Requests\Good;
    use Lumpineevill\Request\APIRequest;
    class GetGoodRequest extends APIRequest
    {
    /**
    * Determine if the user is authorized to make this request.
    *
    * @return bool
    */
    public function authorize()
    {
    return true;
    }
    /**
    * Get the validation rules that apply to the request.
    * simple rule below this.
    * required | exists | required_if | email | interger | date| date_format | between | array | json | boolean |
    * max | min | confirmed | differece | alpha | alpha_numeric | acitve(url) | accepted |distinct | file | same
    * size | string | timezone | unique | url | nullable | not_in| digits | digits_between | mime_type | mime_type_by_file_extension
    * in_array | ip_address | image | regex:patten |
    * @return array
    */
    public function rules()
    {
    return [
    # 'id' => 'required|exists:good,id',
    # 'email' => 'unique:good,email',
    ];
    }
    /**
    * If need to overiding message of node
    * @return array
    */
    public function messages()
    {
    return [];
    }
    }
  • request จะทำงานหน้า validate ข้อมูลก่อนที่จะถูกส่งไปประมวลผลที่ controller เรียกอีกอย่างคือ  clean data นั้นเอง ส่วนนี้ช่วยให้ข้อมูลที่เข้าไปทำงานที่ repository ทำให้เรามั่นใจได้ว่าข้อมูลมันจะต้องตรงตามที่เราต้องการแน่ ๆ ถ้าไม่ใช่ก็ให้คืนค่ากลับไปทันทีไม่ต้องไปทำงานถึง Repository ส่วนที่เป็น business logic แล้วอีกอย่างก็ไม่ต้องไปเช็คเงื่อนไขข้างในเกี่ยวกับ ข้อมูลอีกด้วยช่วยให้ลดบรรทัดของ code ไปเยอะมาก ๆ
หลังจากที่ผ่านการ validate data (clean data) เรียบร้อยแล้วข้อมูลจะถูกส่งต่อไปที่ controller ในส่วนของ controller ของ laravel มันมีความสามารถค่อนข้างหลากหลายเช่นใส่ middleware ตรง controller ก็ได้ แต่สำหรับ package : samark/lumpineevill ผมจะไม่ให้ทำอะไรมาก เพียงให้มันเป็นตัวเชื่อมระหว่าง request -> controller -> repository เพียงเท่านั้นเอง code ที่ออกมาจึงไม่มีเงื่อนไขใด ๆทั้งสิ้น ดังตัวอย่างต่อไปนี้
  • <?php
    /**
    * @author samark chaisanguan
    * @email samarkchsngn@gmail.com
    */
    namespace App\Http\Controllers\Good;
    use App\Http\Controllers\Controller;
    use App\Http\Requests\Good\GetGoodRequest;
    use App\Http\Requests\Good\CreateGoodRequest;
    use App\Http\Requests\Good\UpdateGoodRequest;
    use App\Http\Requests\Good\DeleteGoodRequest;
    use App\Repository\Good\GoodRepository;
    class GoodController extends Controller
    {
    /**
    * set good repository
    * @var object
    */
    protected $good;
    /**
    * [__construct description]
    * @param GoodRepository $good [description]
    */
    public function __construct(GoodRepository $good)
    {
    $this->good = $good;
    }
    /**
    * [createGood description]
    * @param GoodCreateRequest $request [description]
    * @return [type] [description]
    */
    public function createGood(CreateGoodRequest $request)
    {
    $query = $this->good->createData($request->all());
    return response()->json($query);
    }
    /**
    * [getGood description]
    * @param GoodGetRequest $request [description]
    * @return [type] [description]
    */
    public function getGoodList(GetGoodRequest $request)
    {
    $query = $this->good->search($request->all())->getData();
    return response()->json($query);
    }
    /**
    * [deleteGood description]
    * @param GoodDeleteRequest $request [description]
    * @return [type] [description]
    */
    public function deleteGood(DeleteGoodRequest $request)
    {
    $query = $this->good->delete($request->all());
    return response()->json($query);
    }
    /**
    * [updateGood description]
    * @param GoodUpdateRequest $request [description]
    * @return [type] [description]
    */
    public function updateGood(UpdateGoodRequest $request)
    {
    $query = $this->good->updateData($request->all());
    return response()->json($query);
    }
    }
  • จะเห็นว่า code ไม่มีการเช็คเงื่อนไขใด ๆ ทั้งสิ้นผมชอบมันตรงที่ดูแล้วมันสะอาด จริง ๆ มันก็ไม่ได้ถึงกับว่าดี เพราะมีบางกรณีที่ต้องการเช็ค ฉะนั้นถ้า API เส้นไหนที่ต้องการเช็คค่อยมาเพิ่มเองทีหลังแต่โดยรายละเอียดของ code ที่ถูกสร้างขึ้นมาแถบจะไม่ต้องแก้เพราะมันสามารถทำงานได้ทันที
ส่วนต่อมาคือ Repository ไฟล์นี้ก็ถูกสร้างขึ้นมาอัตโนมัติจาก samark/lumpineevill เช่นเดียวกันหน้าที่ของมันคือเชื่อมโยงระหว่าง controll -> repository -> model , business logic(business logic คือ กระบวนการประมวลผลที่ต้องมีการตรวจสอบเงื่อนไขต่าง ๆ ของข้อมูล การจัดการข้อมูลเป็นต้น) ต่าง ๆจะถูกนำมาวางไว้ที่นี้เพื่อลดความซ้ำซ้อนนั่นเอง (ความซ้ำซ้อนคือการเขียน code ชุดเดิม copy ไปไว้ที่อื่น ตอนจะแก้ต้องแก้หลาย ๆ ที่) ตัวอย่างไฟล์ repository ที่ถูกสร้างขึ้นมาด้วย samark/lumpineevill 
  • <?php
    /**
    * @author samark chaisanguan
    * @email samarkchsngn@gmail.com
    */
    namespace App\Repository\Good;
    use App\Models\Good\Good;
    use Lumpineevill\BaseRepository;
    use App\Models\Good\GoodModel;
    class GoodRepository extends BaseRepository implements GoodInterface
    {
    /**
    * set limit
    * @var integer
    */
    protected $limit = 30;
    /**
    * set order by column
    * @var string
    */
    protected $orderBy = 'id';
    /**
    * set sort type
    * @var string
    */
    protected $sortType = 'desc';
    /**
    * set model
    * @var class
    */
    protected $classModel = GoodModel::class;
    /**
    * set search column like
    * @var array
    */
    protected $searchColumnLike = [];
    /**
    * set serach column date
    * @var array
    */
    protected $searchColumnDate = [];
    public function __construct()
    {
    parent::__construct();
    }
    /**
    * GetGoodByID
    * @param $id integer
    * @return mixed
    */
    public function getGoodByID($id)
    {
    return $this->model->find($id);
    }
    }
  •  ส่วนของ repository พื้นฐานที่ถูกสร้างขึ้นมาจะเห็นว่ามีบรรทัดค่อนข้างน้อย แต่ความเป็นจริงแล้วความซ้ำซ้อนทั้งหมดผมนำไปไว้ที่ BaseRepository ทำให้ class ที่ extends (แตก) ออกมาจาก class BaseRepository มันไม่จำเป็นต้องเขียนยาวแต่มีคุณสมบัติเหมือนกับแม่มันเดะคุณสมบัตินี้เรียกว่า  การสืบทอด(อำนาจ) ในทฤษฏีของ OOP (object oriented programming) นั่นเอง
เอาละต่อมาจะมาลองใช้งาน samark/lumpinee กับ laravel 5.* กันสิ่งที่ต้องเตรียมมีดังนี้ ปล.ขั้นตอนการติดตั้ง laravel framwork ผมไม่ขอลงรายเอียดสามารถกลับไปอ่านได้ในบทความเก่า เกี่ยวกับการติดตั้ง laravel ได้ครับ
  • laravel 5.*  [composer create-project laravel/laravel {project-name}]
หลังจากที่เรามี laravel framwork เรียบร้อยแล้วสิ่งต่อมาคือติดตั้ง samark/lumpineevill ด้วย composer ดังนี้
  • composer require samark/lumpineevill
หลังจากที่ติดตั้ง samark/lumpineevill เสร็ดแล้ว ขั้นตอนต่อให้เพิ่มบรรทัดที่ config/app.php ในส่วนของ array providers ดังนี้
  • Lumpineevill\ServiceProvider\LumpineevillServiceProvider::class,
ถ้าไม่มีอะไรผิดพลาดลองพิมพ์คำสั่ง [ php artisan ] มันจะมีคำสั่งหนึ่งเพิ่มเข้ามาคือ samark:genfile 
  • samark
    • samark:genfile     Generate file
ยินดีด้วยคุณมาถูกทางแล้ว ! ต่อมาลองสร้างไฟล์ด้วยคำส่ง php artisan samark:genfile หลังจากพิมพ์เสร็ดเคาะ enter หนึ่งทีจะมีคำถามว่า  
  • What is your namespace ?: >
  • ให้ใส่ชื่อ module ที่คุณต้องการสร้างได้เลย คำแนะนำเพิ่มเติมให้พิมพ์เป็น camel case จะเป็นวิธีที่ดีที่สุด
  • สำหรับตัวอย่างนี้ผมจะใส่ blog เข้าไป

สิ่งแรกที่จะต้องเข้าไปแก้คือ migration ไฟล์ คือคำสั่งที่พิมพ์ไปเมื่อกี้มันสร้าง migration ไฟล์ขึ้นมาให้ด้วย ไปที่ database/migration/create_table_blog_{datetime}.php ตัวอย่างไฟล์
  • <?php
    use Illuminate\Database\Migrations\Migration;
    use Illuminate\Database\Schema\Blueprint;
    use Illuminate\Support\Facades\Schema;
    class CreateTableBlog extends Migration
    {
    /**
    * Run the migrations.
    *
    * @return void
    */
    public function up()
    {
    Schema::create('blog', function (Blueprint $table) {
    $table->increments('id');
    $table->string('topic');
    $table->text('detail');
    $table->timestamps();
    });
    }
    /**
    * Reverse the migrations.
    *
    * @return void
    */
    public function down()
    {
    Schema::dropIfExists('blog');
    }
    }
หลังจากนั้นให้ไปที่ app/Models/Blog/BlogModel.php เพื่อแก้ไข column ให้มันตรงกับไฟล์ migration ดังนี้
  • <?php
    /**
    * @author samark chaisanguan
    * @email samarkchsngn@gmail.com
    */
    namespace App\Models\Blog;
    use Illuminate\Database\Eloquent\Model;
    # use Illuminate\Database\Eloquent\SoftDeletes;
    class BlogModel extends Model
    {
    # use SoftDeletes;
    protected $fillable = [
    'id',
    'topic',
    'detail',
    'created_at',
    'updated_at',
    ];
    /**
    * set table name
    * @var string
    */
    protected $table = 'blog';
    /**
    * set casting value
    * @var array
    */
    protected $casts = [];
    /**
    * set append attribute
    * use get{NameColumn}Attribute function
    * @var array
    */
    protected $appends = [];
    /**
    * set datetime column
    * @var array
    */
    protected $dates = ['created_at', 'updated_at'];
    }
    view raw BlogModel.php hosted with ❤ by GitHub
เมื่อแก้ไขไฟล์เรียบร้อยให้ไปที่ console เพื่อรันคำสั่ง php artisan migrate สั่ง migrate ไฟล์เพื่อสร้าง table ที่ blog ขึ้นมา ถ้าไม่มีอะไรผิดจะขึ้นข้อความดังนี้
  • Migrated: 2018_05_31_012157_create_table_blog
หลังจากนั้นลองย้อนไปที่ routes/api.php จะพบว่า samark/lumpineevill ได้เพิ่ม route ให้แล้วเรียบร้อยดังนี้
  • <?php
    use Illuminate\Http\Request;
    /*
    |--------------------------------------------------------------------------
    | API Routes
    |--------------------------------------------------------------------------
    |
    | Here is where you can register API routes for your application. These
    | routes are loaded by the RouteServiceProvider within a group which
    | is assigned the "api" middleware group. Enjoy building your API!
    |
    */
    Route::get('/user', function (Request $request) {
    return $request->user();
    })->middleware('auth:api');
    # Blog
    require app_path() . '/Http/Routes/API/Blog/BlogRoute.php';
    view raw apiRoute.php hosted with ❤ by GitHub
อีกจุดหนึ่งที่จะลืมแก้ไม่ได้เด็ด ! คือ request เพื่อกรองข้อมูลก่อนส่งไปยัง controller นั่นเอง ซึ่งตัว blog เองผมต้องการข้อมูลอยู่สองอย่างเดียวกันคือ
  1. topic 
  2. detail 
ดังนั้นตัว request ในการสร้างข้อมูลจึงจะต้อง required ข้อมูลเหล่านี้ดังตัวอย่างต่อไปนี้ 
  • <?php
    /**
    * @author samark chaisanguan
    * @email samarkchsngn@gmail.com
    */
    namespace App\Http\Requests\Blog;
    use Lumpineevill\Request\APIRequest;
    class CreateBlogRequest extends APIRequest
    {
    /**
    * Determine if the user is authorized to make this request.
    *
    * @return bool
    */
    public function authorize()
    {
    return true;
    }
    /**
    * Get the validation rules that apply to the request.
    * simple rule below this.
    * required | exists | required_if | email | interger | date| date_format | between | array | json | boolean |
    * max | min | confirmed | differece | alpha | alpha_numeric | acitve(url) | accepted |distinct | file | same
    * size | string | timezone | unique | url | nullable | not_in| digits | digits_between | mime_type | mime_type_by_file_extension
    * in_array | ip_address | image | regex:patten |
    * @return array
    */
    public function rules()
    {
    return [
    'topic' => 'required',
    'detail' => 'required',
    ];
    }
    /**
    * If need to overiding message of node
    * @return array
    */
    public function messages()
    {
    return [];
    }
    }
ทีนี้มาลองยิงสร้างข้อมูลด้วย postman กรณีแรกจะไม่ใส่ข้อมูลใด ๆ ไปถ้ามันถูกต้องต้อง return bad request กลับมา ดังภาพ

ที่นี้มาลองกรณีที่สร้างสำเร็จโดยใส่ข้อมูลของ topic และ detail แล้วส่งไปที่ API ด้วย  postman ดังภาพ

ต่อมาลองทดสอบเรียกข้อมูลที่ถูกสร้างไปเมื่อสักครู่ ด้วย postman ดังภาพ
สำหรับ samark/lumpineevill ตัวนี้มีความสามารถในการ search column เรียกได้ว่าครบทุก column ที่มีอยู่ใน table โดยที่ไม่ต้องแก้ API อะไรเลยมันสะดวกมาก ๆ

สรุป ข้อดีข้อเสียของ samark/lumpineevill
# ข้อดี

  • เป็นเครื่องมือที่สร้าง API ได้ไวที่สุดบน Repository patten 
  • ใช้งานง่าย ติดตั้งง่ายด้วย composer 
  • เหมาะสำหรับผู้เริ่มต้นเขียน API ใหม่และอยากมาลองให้ Repository patten
  • ไฟล์งานแยกเป็นส่วน ๆ ทำให้หาง่าย
# ข้อเสีย 
  • มีไฟล์บางไฟล์ที่ไม่ต้องการถูกสร้างขึ้นมาด้วย
  • API response ไม่ใช่  standard 
  • มีไฟล์ค่อนข้างเยอะ
สุดท้ายอยากให้ลองนำไปใช้กันดูครับ ดีไม่หรือไม่ดียังไงลอง  feed back กลับมาได้ที่ github ของ repo ได้เลย
ปล.งานชิ้นนี้ทำไว้เพื่อสร้างแรงบันดาลใจในการสร้างงานชิ้นถัด ๆ ไป 
ด้วยความที่ผมเบื่อกับการ copy ด้วยมือผมจึงสร้างเครื่องมือนี้ขึ้นมาไม่ต้องกังวลว่าจะทำไฟล์ครบหรือไม่อีกต่อไป. 
ขอบคุณที่อ่านจนจบนะครับ ...
Share on Google Plus

About maxcom

This is a short description in the author block about the author. You edit it by entering text in the "Biographical Info" field in the user admin panel.
    Blogger Comment
    Facebook Comment

0 ความคิดเห็น:

แสดงความคิดเห็น