Skip to content

Instantly share code, notes, and snippets.

@syedanam
Created November 10, 2017 17:41
Show Gist options
  • Save syedanam/bd3324e063a59cb47242cc57edd3a7d3 to your computer and use it in GitHub Desktop.
Save syedanam/bd3324e063a59cb47242cc57edd3a7d3 to your computer and use it in GitHub Desktop.
Laravel Facade class
লারাভেল Facade কি? কিভাবে নিজস্ব Facade তৈরি করা যায়?
Written 2 days ago by Al Imran Ahmed on PHP
ভূমিকার আগে
আমি এই লিখায় লারাভেলের 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 কোড রয়ে গিয়েছিল। এই ব্যাপারটা যারা যারা আমাকে জানিয়েছেন তাদের সবাইকে ধন্যবাদ। আমি কোড ঠিক করে দিয়েছি।
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment