How to structure a modular app in Laravel 5?

PhpLaravelLaravel RoutingLaravel 5

Php Problem Overview


I would like to divide my application in modules. For instance, there would be a "core" modules that contains the basic login functionality, app layout/formatting (CSS etc), user management and a diary.

Later on I may create other modules like a contact manager that can easily be added or removed from the application.

There would be some logic in the apps navigation for determining which modules are present and to show/hide the links to them.

How can I do this in terms of directory structure, namespaces and anything else that's needed?


I am looking at creolab/laravel-modules but it states that it is for Laravel 4. Can I still use it with 5 in exactly the same way?

The documentation says to place models, controllers and views within each module directory, but how does this work with routes? Ideally I would like each module to have its own routes.php file. How will all of this work with the stuff in the http and the resources directory?


I was thinking of something like this:

Module idea

But I have no idea how I would get it to work.


I have just tried the tutorial here:

http://creolab.hr/2013/05/modules-in-laravel-4/

With no extra libraries etc, just pure Laravel 5.

I seem to have hit a brick wall with an error message:

FatalErrorException in ServiceProvider.php line 16:
Call to undefined method Illuminate\Config\Repository::package()

Regarding the following:

<?php namespace App\Modules;

abstract class ServiceProvider extends \Illuminate\Support\ServiceProvider
{

    public function boot()
    {
        if ($module = $this->getModule(func_get_args())) {
            $this->package('app/' . $module, $module, app_path() . '/modules/' . $module);
        }
    }

    public function register()
    {
        if ($module = $this->getModule(func_get_args())) {
            $this->app['config']->package('app/' . $module, app_path() . '/modules/' . $module . '/config');

// Add routes
            $routes = app_path() . '/modules/' . $module . '/routes.php';
            if (file_exists($routes)) require $routes;
        }
    }

    public function getModule($args)
    {
        $module = (isset($args[0]) and is_string($args[0])) ? $args[0] : null;

        return $module;
    }

}

What is causing this and how can I fix it?


Got my head around this a bit more now. Got my package/module routes and views working which is great:

abstract class ServiceProvider extends \Illuminate\Support\ServiceProvider
{

    public function boot()
    {
        if ($module = $this->getModule(func_get_args())) {
            include __DIR__.'/'.$module.'/routes.php';
        }
        $this->loadViewsFrom(__DIR__.'/'.$module.'/Views', 'core');
    }

    public function register()
    {
        if ($module = $this->getModule(func_get_args())) {

        }
    }

    public function getModule($args)
    {
        $module = (isset($args[0]) and is_string($args[0])) ? $args[0] : null;

        return $module;
    }

}

I have one last question, how would I load all my controllers from inside my package, much like how the loadViewsFrom() method works?

Php Solutions


Solution 1 - Php

I seem to have figured it all out.

I'll post it here in case it helps other beginners, it was just about getting the namespaces right.

In my composer.json I have:

...
"autoload": {
	"classmap": [
		"database",
        "app/Modules"
	],
	"psr-4": {
		"App\\": "app/",
        "Modules\\": "Modules/"
	}
}

My directory and files ended up like this:

enter image description here

I got my Core module router.php to work by wrapping my controllers for that module in a group specifying the namespace:

Route::group(array('namespace' => 'Modules\Core'), function() {
    Route::get('/test', ['uses' => 'TestController@index']);
});

I imagine when I come to doing my models for the package it will be a similar case of getting the namespaces right.

Thanks for all your help and patience!

Solution 2 - Php

Solution:

> Step1: Create Folder “Modules” inside “app/”


> Step2: In Modules folder create your Module (Module1( suppose admin Module))

 Inside admin module : create the following folder 

 1. Controllers  (here will your controller files)
 2. Views  (here will your View files)
 3. Models  (here will your Model files)
 4. routes.php (here will your route code in this file)

Similarly, you can create multiple modules

Module2( suppose API )
-Controllers
-Views
-Models
-routes.php

> Step3 : Create ModulesServiceProvider.php inside “Modules/” Folder


> Step4 : Paste following code inside ModulesServiceProvider.php

<?php

namespace App\Modules;

/**
 * ServiceProvider
 *
 * The service provider for the modules. After being registered
 * it will make sure that each of the modules are properly loaded
 * i.e. with their routes, views etc.
 *
 * @author kundan Roy <query@programmerlab.com>
 * @package App\Modules
 */

use Illuminate\Support\Facades\Route;
use Illuminate\Foundation\Support\Providers\RouteServiceProvider as ServiceProvider;

class ModulesServiceProvider extends ServiceProvider {

	/**
	 * Will make sure that the required modules have been fully loaded
	 *
	 * @return void routeModule
	 */
	public function boot() {
		// For each of the registered modules, include their routes and Views
		$modules=config("module.modules");

		while (list(,$module)=each($modules)) {

			// Load the routes for each of the modules

			if (file_exists(DIR.'/'.$module.'/routes.php')) {

				include DIR.'/'.$module.'/routes.php';
			}

			if (is_dir(DIR.'/'.$module.'/Views')) {
				$this->loadViewsFrom(DIR.'/'.$module.'/Views',$module);
			}
		}
	}

	public function register() { }

}

> Step5 : Add following line inside ‘config/app.php’ file

App\Modules\ModulesServiceProvider::class,

> Step6 : Create module.php file inside ‘config’ folder > > Step7 : Add following code inside module.php (path => “config/module.php”)

<?php

return [
	'modules'=>[
		'admin',
		'web',
		'api'
	]
];

Note : You can add your module name whichever you have created. Here there are modules.

> Step8 : Run this command

composer dump-autoload

Solution 3 - Php

A little late, but if you want to use modules in your future projects, i've written a module generator. It generates modules via php artisan make:module name You can also just drop some modules in the app/Modules folder and they are ready to use/work. Take a look. Save some time ;)

l5-modular

Solution 4 - Php

You can also use pingpong-labs

documentations Here.

Here is an example.

You can just install and check the process.

Note: I am not advertising. Just checked that cms built on Laravel with module support. So thought that might be helpful for you and others.

Solution 5 - Php

Kundan roy: I liked your solution but I copied your code from StackOverflow, I had to change the quotes and semi-quotes to get it working - I think SOF replace these. Also changed Dir for base_path() to be more inline with Laravel's (new) format.

namespace App\Modules;

/**
* ServiceProvider
*
* The service provider for the modules. After being registered
* it will make sure that each of the modules are properly loaded
* i.e. with their routes, views etc.
*
* @author kundan Roy <query@programmerlab.com>
* @package App\Modules
*/

use Illuminate\Support\Facades\Route;
use Illuminate\Foundation\Support\Providers\RouteServiceProvider as ServiceProvider;

class ModulesServiceProvider extends ServiceProvider
{

/**
* Will make sure that the required modules have been fully loaded
* @return void routeModule
*/
   public function boot()
{
    // For each of the registered modules, include their routes and Views
    $modules = config("module.modules");

    while (list(,$module) = each($modules)) {
   
        // Load the routes for each of the modules
        if(file_exists(base_path('app/Modules/'.$module.'/routes.php'))) {
            include base_path('app/Modules/'.$module.'/routes.php');
        }

        // Load the views                                           
        if(is_dir(base_path('app/Modules/'.$module.'/Views'))) {
            $this->loadViewsFrom(base_path('app/Modules/'.$module.'/Views'), $module);
        }
    }
}

public function register() {}

}

Solution 6 - Php

pingpong/modules is a laravel package which created to manage your large laravel app using modules. Module is like a laravel package for easy structure, it have some views, controllers or models.

It's working in both Laravel 4 and Laravel 5.

To install through composer, simply put the following in your composer.json file:

{
    "require": {
        "pingpong/modules": "~2.1"
    }
}

And then run composer install to fetch the package.

To create a new module you can simply run :

php artisan module:make <module-name>

- Required. The name of module will be created. Create a new module

php artisan module:make Blog

Create multiple modules

php artisan module:make Blog User Auth

for more visit: https://github.com/pingpong-labs/modules

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
Questionimperium2335View Question on Stackoverflow
Solution 1 - Phpimperium2335View Answer on Stackoverflow
Solution 2 - PhpKundan royView Answer on Stackoverflow
Solution 3 - PhpGordon FreemanView Answer on Stackoverflow
Solution 4 - PhpJnanaranjanView Answer on Stackoverflow
Solution 5 - PhpgabrielkolbeView Answer on Stackoverflow
Solution 6 - PhpSagar NaliyaparaView Answer on Stackoverflow