- بواسطة x32x01 ||
لو اشتغلت قبل كده على مشروع كبير باستخدام Laravel، أكيد قابلت موقف غريب شوية…
تفتح ملف Service علشان تعدّل حاجة بسيطة، تلاقيه بينادي Service تاني… والـ Service التاني بينادي Service ثالث 😅
وفجأة تلاقي نفسك تايه بين الملفات ومش فاهم فين أصل المشكلة.
على الورق، الموضوع بيبان كأنه Architecture نظيفة ومنظمة.
لكن في الواقع أحيانًا بيكون العكس تمامًا.
في البوست ده هنتكلم عن تجربة واقعية مع تنظيم الكود في Laravel، وامتى فعلًا نحتاج Service، وامتى استخدامه بيزود التعقيد بدل ما يحل المشكلة.
النتيجة غالبًا بتكون:
المشكلة إن التنظيم ده أحيانًا بيكون مبالغ فيه وبيخلق تعقيد غير ضروري.
مثال:
في الحالة دي غالبًا:
النتيجة؟
مثال بسيط:
كل Service بينادي Service تاني.
النتيجة:
Bug بسيط ممكن يخليك تتنقل بين:
وده بيأثر جدًا على سرعة تطوير المشروع.
الفكرة الأساسية: لا تستخدم Service إلا عندما تحتاجه فعلًا.
لكن في الحقيقة Controller ممكن يحتوي على:
الكود واضح ومباشر، بدون طبقات إضافية غير ضرورية.
استخدمه عندما يكون هناك:
في الحالة دي وجود Service منطقي جدًا.
لكن مهم جدًا إن Repository يحتوي على:
بكده بيكون دور Repository واضح ومحدد.
هل المنطق:
يبقى Service مناسب جدًا.
لكن لو المنطق بسيط وموجود في Controller مرة واحدة فقط…
الأفضل إنك تخليه مباشر وبسيط.
كل طبقة إضافية في المشروع تعني:
لكن استخدامه بشكل مفرط ممكن يحول المشروع إلى شبكة معقدة من الطبقات.
القاعدة البسيطة: Service بدون سبب = عبء صيانة.
اجعل الكود واضح، بسيط، وسهل الفهم لأي مطور يفتح المشروع بعدك.
إيه الطبقة اللي حسيت إنها كانت سبب التعقيد الحقيقي في المشروع؟
تفتح ملف Service علشان تعدّل حاجة بسيطة، تلاقيه بينادي Service تاني… والـ Service التاني بينادي Service ثالث 😅
وفجأة تلاقي نفسك تايه بين الملفات ومش فاهم فين أصل المشكلة.
على الورق، الموضوع بيبان كأنه Architecture نظيفة ومنظمة.
لكن في الواقع أحيانًا بيكون العكس تمامًا.
في البوست ده هنتكلم عن تجربة واقعية مع تنظيم الكود في Laravel، وامتى فعلًا نحتاج Service، وامتى استخدامه بيزود التعقيد بدل ما يحل المشكلة.
المشكلة الشائعة في مشاريع Laravel الكبيرة
كتير من المطورين بيحاولوا يطبقوا Clean Architecture أو Service Layer Pattern بشكل حرفي جدًا.النتيجة غالبًا بتكون:
- Controller بسيط جدًا
- Service لكل عملية
- Repository للوصول إلى قاعدة البيانات
Code:
Controller
↓
Service
↓
Repository
↓
Database عندما يتحول Service Layer إلى عبء
خلينا نشوف بعض الحالات اللي بتحصل كتير في مشاريع Laravel.Service لكل عملية CRUD
واحدة من أشهر الأخطاء إن كل عملية CRUD تبقى لها Service مستقل.مثال:
Code:
CreateUserService
UpdateUserService
DeleteUserService - نفس المنطق تقريبًا
- نفس البيانات
- اختلاف بسيط جدًا
النتيجة؟
- التنقل بين الملفات بقى أبطأ
- قراءة الكود أصعب
- الصيانة أصعب
Service داخل Service
مشكلة تانية شائعة جدًا هي تداخل الخدمات.مثال بسيط:
Code:
UserService
→ OrderService
→ WalletService النتيجة:
- Trace طويل جدًا
- صعوبة تتبع تدفق التنفيذ
- Debug مرهق
Debug أصبح كابوس
المشكلة الأكبر بتظهر وقت اكتشاف الأخطاء (Debugging).Bug بسيط ممكن يخليك تتنقل بين:
- Controller
- Service
- Service تاني
- Repository
- Model
- Helper
وده بيأثر جدًا على سرعة تطوير المشروع.
الحل الواقعي لتنظيم الكود في Laravel
بعد تجارب كتير في مشاريع Laravel، في طريقة أبسط بكتير وبتحافظ على وضوح الكود.الفكرة الأساسية: لا تستخدم Service إلا عندما تحتاجه فعلًا.
اجعل Controller واضحًا وليس فارغًا
بعض المطورين بيحاولوا يخلو Controller شبه فارغ.لكن في الحقيقة Controller ممكن يحتوي على:
- Validation
- Decision logic بسيط
- استدعاء مباشر للعمليات
Code:
public function store(Request $request)
{
$data = $request->validate([
'name' => 'required',
'email' => 'required|email'
]);
$user = User::create($data);
return response()->json($user);
} استخدم Service فقط عندما يكون هناك منطق فعلي
الـ Service Layer في Laravel مفيد جدًا… لكن بشرط.استخدمه عندما يكون هناك:
- Business Logic حقيقي
- عمليات مركبة
- منطق بيتكرر في أماكن مختلفة
PHP:
class TransferBalanceService
{
public function transfer($fromUser, $toUser, $amount)
{
$fromUser->wallet -= $amount;
$toUser->wallet += $amount;
$fromUser->save();
$toUser->save();
}
} استخدم Repository فقط للوصول إلى البيانات
الـ Repository Pattern مفيد عندما تريد فصل الوصول إلى البيانات عن باقي الكود.لكن مهم جدًا إن Repository يحتوي على:
- Queries فقط
- بدون Business Logic
PHP:
class UserRepository
{
public function findByEmail($email)
{
return User::where('email', $email)->first();
}
} متى تحتاج Service فعلًا في Laravel؟
اسأل نفسك السؤال ده قبل إنشاء Service جديد.هل المنطق:
✔ معقد؟
✔ يتكرر في أكثر من مكان؟
✔ مستقل عن HTTP؟
✔ ممكن يستخدم في Job أو Command أو Listener؟
لو الإجابة نعم…يبقى Service مناسب جدًا.
لكن لو المنطق بسيط وموجود في Controller مرة واحدة فقط…
الأفضل إنك تخليه مباشر وبسيط.
القاعدة الذهبية في تصميم مشاريع Laravel
التعقيد مش دليل احتراف. أحيانًا أفضل Architecture هي الأبسط.كل طبقة إضافية في المشروع تعني:
- ملفات أكثر
- وقت Debug أطول
- صيانة أصعب
خلاصة التجربة
استخدام Service Layer في Laravel مفيد جدًا عندما يكون هناك منطق حقيقي يحتاج تنظيم.لكن استخدامه بشكل مفرط ممكن يحول المشروع إلى شبكة معقدة من الطبقات.
القاعدة البسيطة: Service بدون سبب = عبء صيانة.
اجعل الكود واضح، بسيط، وسهل الفهم لأي مطور يفتح المشروع بعدك.
سؤال للمطورين
لو اشتغلت قبل كده على مشروع Laravel كبير 👨💻إيه الطبقة اللي حسيت إنها كانت سبب التعقيد الحقيقي في المشروع؟
- Service Layer؟
- Repository Pattern؟
- Controllers؟
- ولا حاجة تانية؟
التعديل الأخير: