লারাভেল Facade কি? কিভাবে নিজস্ব Facade তৈরি করা যায়?

Published on PHP by Al Imran Ahmed

ভূমিকার আগে

আমি এই লিখায় লারাভেলের Facade কি এবং কিভাবে তার উত্তর দিতে চেষ্টা করব। তবে কেউ যদি জানতে চান Facade কেন তাহলে একটু কষ্ট করে লারাভেলের ডকুমেন্টেশনের Facade-সেকশনটা দেখে নিতে পারেন। যদি ডকুমেন্টেশনের কোন অংশ বুঝতে অসুবিধা হয় তাহলে নিঃসংকোচে আমাকে প্রশ্ন করতে পারেন। আমি আজকের লিখায় লারাভেল Facade কিভাবে কাজ করে? কিভাবে নিজের মত করে নিজের একটি Facade বানিয়ে ফেলা যায়? এই সব প্রশ্নের উত্তর দেওয়ার চেষ্টা করব এই লিখায়।

ভূমিকা

লারাভেলে কাজ করার শুরুর দিকে আমি Facade-কে ভাবতাম এমন একটা ক্লাস যার মধ্যে কতগুলো স্টেটিক ম্যাথড আছে। এর পিছনে কারণ হচ্ছে লারাভেল যারা শুরু করে তাদের কাছে লারাভেলের এপ্লিকেশনে ঢোকার রাস্তা(Entry point) হলো রুট(route.php বর্তমানে web.php, api.php ইত্যাদি) ফাইল। আর সেই রুট ফাইলে আমরা কোন রুট লিখি এভাবে:

Route::get('home', 'HomeController@show');

কেবল মাত্র প্রাথমিক অবজেক্ট অরিয়েন্টেড জ্ঞান সম্পন্ন যে কোন বিশ্বাবিদ্যালয় পড়ুয়া প্রোগ্রামার উপরের এই লাইনটা দেখে এটা ভাবাই স্বাভাবিক যে, নিশ্চই Route নামে একটা ক্লাস আছে যার মধ্যে get নামে একটা স্টেটিক মেথড আছে। আর এখানে সেই মেথডটিতে দুইটা আরগুমেন্ট পাঠানো হচ্ছে। অন্য কেউ শুরুর দিকে ভাবুব-না ভাবুক আমি অন্তত্য তাই ভাবছিলাম। তো কিছুদিন পর আমি কৌতুহলবসত দেখতে চাইলাম যে এই get মেথডটিতে কি লিখা আছে। কিন্তু আমি আমার কোড এডিটর থেকে যখন ঐ Route ক্লাসটিতে গেলাম তখন বেশ বড় ধরনের একটা ধাক্কা খেলাম কারণ ঐ ক্লাসে get নামে কোন মেথড নেই! তাহলে কি আমি অবজেক্ট অরিয়েন্টেড প্রোগ্রামিং-এর এমন সাধারণ বিষয়ও ঠিকমত জানি না? পরে জানতে পারলাম এই Route ক্লাসটি লারাভেলের এক ধরনের বিশেষ ক্লাস যাদেরকে আসলে Facade বলা হয়। এই Facade নিয়েই আজকের লিখা।
পরবর্তিতে, আমি যখন আমাদের কোম্পানিতে নিয়োগের জন্য লারাভেলে অভিজ্ঞতাসম্পন্ন বিভিন্ন প্রোগ্রামাদের সাক্ষাৎকার নিতাম তখন এই প্রশ্নটা প্রায়ই করতাম। কিন্তু দুঃখের বিষয় হল, অনেক প্রোগ্রামারকে আমি স্টেটিক মেথড কি এটা নিয়েই দ্বিধায় ভুগতে দেখেছি। আর যাদের স্টেটিক মেথড সম্পর্কে ধারণ আছে তারা এই ব্যাপারে আমার সাথে এক মত হতেন যে, Route ক্লাসের মধ্যে get নামে একটি স্টেটিক মেথড আছে, যা আসলে ভূল। এই ভূল উত্তরটি আমি আনাড়ী লারাভেল প্রোগ্রামার থেকে শুরু করে দুই তিন বছরের অভিজ্ঞতা সম্পন্ন প্রগ্রামারদের কাছেও পেয়েছি, যা আসলে একেবারেই অপ্রত্যাশিত ছিল।

Facade কি?

লারাভেলের Facade হল এমন একটি ক্লাস যে অন্য একটি ক্লাসের স্টেটিক ইন্টারফেইস হিসাবে কাজ করে। আরেকটু সহজভাবে বললে, Facade ক্লাস অন্য একটি ক্লাসের মেথড কে কল করার মাধ্যম হিসাবে কাজ করে। তবে ঐ ক্লাসটির কোন মেথড স্টেটিক মেথড না হওয়া সত্যেও Facade ক্লাস ব্যাবহার করে যখন কল করা হয় স্টেটিক মেথডের মত কল করা যায়। তার মানে, এখন আমরা বলতে পারি, লারাভেলের Facade আসলে অন্য একটি ক্লাসের নন-স্টেটিক মেথডকে স্টেটিক মেথডের মত কল করার মাধ্যম হিসাবে কাজ করে। Route এর মত লারাভেলে আরও অনেক জনপ্রিয় Facade আছে যেমন, View, Session, Cache, Request ইত্যাদি

কিভাবে নিজস্ব Facade বানাতে হয়?

উদাহরণের স্বার্থে মনে করি, আমরা আমাদের লারাভেল এপ্লিকেশনে গানিতিক হিসাব-নিকাশ করার জন্য Math একটি নামে Facade লিখতে চাই। যেই Facade ব্যাবহার করে, Math::add($number1, $number2)Math::sub($number1, $number2) লিখলে যথাক্রমে দুইটি নাম্বারের যোগ ও বিয়োগ করা যাবে। এর জন্য নিচের ৩টি ধাপ অনুসরণ করতে হবেঃ

১। একটি ক্লাস লিখতে হবে যে ক্লাসটি আসলে এই হিসাব নিকাকেশের কাজগুলো করবে। অনেকটা নিম্নরূপঃ

<?php
namespace App\Services;

class Calculator
{
    public function add($number1, $number2){
        return $number1 + $number2;
    }

    public function sub($number1, $number2){
        return $number1 - $number2;
    }
}

২। এবার আমাদের সেই বিখ্যাত Facade ক্লাসটি লিখে ফেলিঃ

<?php
namespace App\Facades;

use Illuminate\Support\Facades\Facade;

class Math extends Facade
{
    protected static function getFacadeAccessor(){
        return 'Math';
    }
}

হ্যা, এই টুকুই, আর কিছু লিখতে হবেনা এই ক্লাসে। এটাই আসল জাদু...

৩। এইবার আমাদের সার্ভিস প্রোভাইডারে `Math` স্ট্রিং-এর সাথে ক্লাসকে বাইন্ড করতে হবে।

<?php 
namespace App\Providers;

use Illuminate\Support\ServiceProvider;
use App\Services\Calculator;

class AppServiceProvider extends ServiceProvider
{
    public function register(){
        $this->app->bind('Math', Calculator::class);
    }
}

এই মানে আমরা `AppServiceProvider` এ বলতেছি, কেউ যদি `Math` স্ট্রিং দিয়ে লারাভেলের সার্ভিস কন্টেইনারকে বলে আমাকে একটা ক্লাস দাও তাহলে সে `Calculator` ক্লাস এর ইন্সটেন্স রিটার্ন করবে। ব্যাস, আমাদের `Facade` তৈরি করা হয়ে গেল! এখন আমরা কোন একটি কন্ট্রোলারে নিচের মত কোড লিখে আমাদের নতুন বানানো `Facade`-টি টেস্ট করতে পারব।

<?php

namespace App\Http\Controllers;

use App\Facades\Math;

class TestController extends Controller
{
    public function testMath(){
        return Math::add(5, 10); //will return 15
    }
}



কিভাবে কাজ করে?

এখন প্রশ্ন হলো উপরে আমরা Facade বানানোর জন্য যে কোড গুলো লিখেছি, সেই কোড গুলো আসলে কিভাবে একটা আরেকটার সাথে সম্পর্কিত? আর এখানে Calculator ক্লাসে কোন স্টেটিক মেথড না থাকা সত্যেও আমরা কিভাবে Math::add(5, 10) কল করতে পারতেছি?
এই প্রশ্ন গুলোর উত্তর খোঁজার জন্য আমরা আমাদের testMath মেথড থেকে শুরু করে যদি পেছন দিকে যায় তাহলে কি দেখতে পাব?
Math::add(5, 10); এর মানে Math Facade এর add নামের স্টেটিক মেথডকে কে কল করা হচ্ছে। এখন আমরা যদি Math Facade-এ যায় তাহলে দেখতে পাব, এর মধ্যে add নামে কোন স্টেটিক মেথড নেই তবে getFacadeAccessor নামে একটি স্টেটিক মেথড আছে যা Math স্ট্রিং রিটার্ন করতেছে। এর মানে এই মেথডকে ক্লাসের কোন স্টেটিক মেথডকে যদি কেউ কল করে তাহলে সে Service Container-এ দেখবে Math স্ট্রিং-এর জন্য কোন ক্লাস বাইন্ড করা আছে কিনা, যদি থাকে তাহলে ঐ ক্লাসের ইন্সটেন্স এর মধ্যে ঐ মেথডটি কল করবে। আর এই পুরো ব্যাপারটি নিয়ন্ত্রন করছে Math ক্লাসের আব্বা Facade ক্লাস। এই জন্য আমরা Math ক্লাসে শুধু এক লাইনের একটা মেথড লিখেছি, আর কিছু করতে হয়নি।
এই ছিল লারাভেল Facade নিয়ে আজকের আলোচনা, এই লম্বা লিখা পড়ার জন্য সবাইকে ধন্যবাদ। কোন প্রশ্ন কিংবা মতামত থাকলে নিঃসংকোচে মন্তব্য করতে পারেন।

আপডেট(০৮-১১-২০১৭)

আমি আসলে প্রথমে ভেবেছিলাম Http নামে একটা Facade বানিয়ে উদাহরণ দিব, তাই Http নামের Facade এর জন্য কোড লিখেছিলাম। কিন্তু Http দিয়ে উদাহরণ দিলে অনেক বিগিনারদের কাছে জটিল মনে হতে পারে ভেবে পরে Math Facade দিয়ে উদাহরণ দিয়েছি। কিন্তু কিছু কিছু জায়গায় আগের Http কোড রয়ে গিয়েছিল। এই ব্যাপারটা যারা যারা আমাকে জানিয়েছেন তাদের সবাইকে ধন্যবাদ। আমি কোড ঠিক করে দিয়েছি।

More articles on PHP

Comments(7)

+

Raktim said

ভাল হইছে... আমাদের এমন অনেক আর্টিকেল দরকার......

Al Imran Ahmed replied

ধন্যবাদ, চেষ্টা করব ইনশাআল্লাহ

Shafiul Alam Biplob said

Nice Article. But Something is wrong with ' Math' and 'Http' term I think. Please double check.

Md. Shamim Shahnewaz said

Http::add($number1, $number2) and Math::add(5, 10) These are getting me confused. Actually you told গানিতিক হিসাব-নিকাশ করার জন্য Math একটি নামে Facade লিখতে চাই। যেই Facade ব্যাবহার করে, Http::add($number1, $number2) ও Http::sub($number1, $number2) লিখলে যথাক্রমে দুইটি নাম্বারের যোগ ও বিয়োগ করা যাবে। But you called Math::add(5, 10) in your controller. My question is what is Http::add() actually? Can we use it in controller instead of Math::add() If not then what is this actually? :)

sabbir ahmed said

1. Class Math should extend Facade class. 2. In AppServiceProvider it have to bind 'Math' with Calculator class (with location)

Al Imran Ahmed replied

@sabbir ahmed, আপনার দুই নম্বর পয়েন্টটা আসলে Calculator হবে, Math হবে না। আর লোকেশন বলতে হয় নি কারণ উপরে use স্টেইট্মেন্ট দিয়ে ঐ ক্লাসকে ব্যাবহার করা হয়েছে।

Rahat said

Laravel এর প্রতিটা কাজে Facade ব্যবহার করেছি কিন্তু Facade এর অর্থ আজকে বুজলাম। ধন্যবাদ।

Al Imran Ahmed said

আমি আসলে প্রথমে ভেবেছিলাম Http নামে একটা Facade বানিয়ে উদাহরণ দিব, তাই Http নামের Facade এর জন্য কোড লিখেছিলাম। কিন্তু Http দিয়ে উদাহরণ দিলে অনেক বিগিনারদের কাছে জটিল মনে হতে পারে ভেবে পরে Math Facade দিয়ে উদাহরণ দিয়েছি। কিন্তু কিছু কিছু জায়গায় আগের Http কোড রয়ে গিয়েছিল। এই ব্যাপারটা যারা যারা আমাকে জানিয়েছেন তাদের সবাইকে ধন্যবাদ। আমি কোড ঠিক করে দিয়েছি।

Rakib Hasan Sabbir said

Dear via, if you include, why to use or why needed(because many more solutions are there) types of a section it could be more helpful I think.
No noise, unsubscribe anytime!
© 2021 Al Imran Ahmed
Contact