feat-wip(ISPConfig & Dolibarr connections)
This commit is contained in:
18
.env.example
18
.env.example
@@ -63,3 +63,21 @@ AWS_BUCKET=
|
|||||||
AWS_USE_PATH_STYLE_ENDPOINT=false
|
AWS_USE_PATH_STYLE_ENDPOINT=false
|
||||||
|
|
||||||
VITE_APP_NAME="${APP_NAME}"
|
VITE_APP_NAME="${APP_NAME}"
|
||||||
|
|
||||||
|
DOLIBARR_URL=
|
||||||
|
LIBARR_USERNAME=
|
||||||
|
DOLIBARR_PWD=
|
||||||
|
|
||||||
|
MAIL_ISPAPI_URL=
|
||||||
|
MAIL_ISPAPI_USERNAME=
|
||||||
|
MAIL_ISPAPI_PWD=
|
||||||
|
|
||||||
|
HOSTING_ISPAPI_URL=
|
||||||
|
HOSTING_ISPAPI_USERNAME=
|
||||||
|
HOSTING_ISPAPI_PWD=
|
||||||
|
|
||||||
|
#NEXTCLOUD_API_ID=
|
||||||
|
#NEXCLOUD_API_PWD=
|
||||||
|
|
||||||
|
#SYMPA_API_ID=
|
||||||
|
#SYMPA_API_PWD=
|
||||||
|
|||||||
11
README.md
11
README.md
@@ -1,15 +1,16 @@
|
|||||||
# LRL APP - Centralized Portal with Laravel, React & Keycloak SSO
|
# LRL APP - Centralized Portal with Laravel, React & Keycloak SSO
|
||||||
|
|
||||||
This project is a **centralized portal application** built with **Laravel 12 & React19**, designed to provide a seamless and secure entry point for both **end users** and **administrators**.
|
This project is a **centralized portal application & web hosting association ERP** built with **Laravel 12 & React19**, designed to provide a seamless and secure entry point for both **end users** and **administrators**.
|
||||||
|
|
||||||
- **Front Office (Users)**
|
- **Front Office (Users)**
|
||||||
- Build on React19
|
- Build on React19
|
||||||
- (V2) Authentication via **Keycloak SSO (OIDC)**
|
- (V2) Authentication via **Keycloak SSO (OIDC)**
|
||||||
- Unified dashboard to access external applications (cloud storage, mailing tools, file sharing, etc.)
|
- Unified dashboard to access external applications (cloud storage, mailing tools, file sharing, etc.)
|
||||||
- Role-based access control synced from Keycloak
|
- Role-based access control synced from Keycloak
|
||||||
|
- Connected with : ISP Config for web hosting and mailbox management, NextCloud, Sympa for mailing list and more...
|
||||||
- **Back Office (Admins)**
|
- **Back Office (Admins)**
|
||||||
- Authentication handled **locally in Laravel** (separate from Keycloak)
|
- Authentication handled **locally in Laravel** (separate from Keycloak)
|
||||||
- Built with **FilamentPHP** for a modern and intuitive admin panel
|
- Built with **FilamentPHP**
|
||||||
- Advanced admin features: app configuration, user activity logs, monitoring
|
- Advanced admin features: app configuration, user activity logs, monitoring
|
||||||
- **Security & API**
|
- **Security & API**
|
||||||
- JWT validation for user-facing APIs (via Keycloak)
|
- JWT validation for user-facing APIs (via Keycloak)
|
||||||
@@ -20,10 +21,10 @@ This project is a **centralized portal application** built with **Laravel 12 & R
|
|||||||
- Blade + Livewire (back office UI)
|
- Blade + Livewire (back office UI)
|
||||||
- React19 (front office UI)
|
- React19 (front office UI)
|
||||||
- TailwindCSS (UI framework)
|
- TailwindCSS (UI framework)
|
||||||
- Keycloak SSO (OIDC)
|
- Keycloak SSO (OIDC) (V2)
|
||||||
- FilamentPHP (admin panel)
|
- FilamentPHP (admin panel)
|
||||||
- Redis (cache, sessions, queues)
|
- Redis (cache, sessions, queues)
|
||||||
- MySQL (PostgreSQL coming soon)
|
- Maria DB
|
||||||
- Docker-ready + CI/CD support (coming soon)
|
- Docker-ready + CI/CD support and automated deploy
|
||||||
|
|
||||||
This architecture allows associations to **centralize authentication and app access** while keeping the **admin back office independent and highly secure**.
|
This architecture allows associations to **centralize authentication and app access** while keeping the **admin back office independent and highly secure**.
|
||||||
|
|||||||
86
app/Console/Commands/SyncDolibarrMembers.php
Normal file
86
app/Console/Commands/SyncDolibarrMembers.php
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Console\Commands;
|
||||||
|
|
||||||
|
use App\Models\Member;
|
||||||
|
use App\Services\DolibarrService;
|
||||||
|
use Carbon\Carbon;
|
||||||
|
use Illuminate\Console\Command;
|
||||||
|
use Illuminate\Http\Client\ConnectionException;
|
||||||
|
use function Deployer\timestamp;
|
||||||
|
|
||||||
|
class SyncDolibarrMembers extends Command
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* The name and signature of the console command.
|
||||||
|
*
|
||||||
|
* @var string
|
||||||
|
*/
|
||||||
|
protected $signature = 'app:sync-dolibarr-members';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The console command description.
|
||||||
|
*
|
||||||
|
* @var string
|
||||||
|
*/
|
||||||
|
protected $description = 'Retrieve members data from Dolibarr';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute the console command.
|
||||||
|
* @throws ConnectionException
|
||||||
|
*/
|
||||||
|
public function handle(): void
|
||||||
|
{
|
||||||
|
$this->info('Starting Dolibarr members import...');
|
||||||
|
// Dolibarr API call
|
||||||
|
$client = new DolibarrService;
|
||||||
|
$doliMembers = collect($client->getAllMembers());
|
||||||
|
|
||||||
|
$progressBar = $this->output->createProgressBar(count($doliMembers));
|
||||||
|
$progressBar->start();
|
||||||
|
$i = 0;
|
||||||
|
|
||||||
|
foreach ($doliMembers as $member) {
|
||||||
|
dd($member);
|
||||||
|
|
||||||
|
$newMember = Member::updateOrCreate([
|
||||||
|
'dolibarr_id' => $member->id,
|
||||||
|
], [
|
||||||
|
'status' => $member['status'], // @todo: faire concorder les statuts
|
||||||
|
'nature' => 'physical',
|
||||||
|
'member_type' => $member['type'],
|
||||||
|
'group_id' => null,
|
||||||
|
'lastname' => $member['firstname'],
|
||||||
|
'firstname' => $member['lastname'],
|
||||||
|
'email' => $member['email'],
|
||||||
|
'personal_email' => '',
|
||||||
|
'company' => '',
|
||||||
|
'website_url' => $member['url'],
|
||||||
|
'date_of_birth' => '',
|
||||||
|
'address' => '',
|
||||||
|
'zipcode' => '',
|
||||||
|
'city' => '',
|
||||||
|
'country' => '',
|
||||||
|
'phone1' => '',
|
||||||
|
'phone2' => '',
|
||||||
|
'public_membership' => '',
|
||||||
|
'created_at' => Carbon::create($member['date_creation'])->format(timestamp()),
|
||||||
|
'updated_at' => '',
|
||||||
|
]);
|
||||||
|
|
||||||
|
// On crée l'adhérent en remplissant les données en bdd avec ses coordonnées, son statut etc ...
|
||||||
|
|
||||||
|
// On récupère toutes les adhésions/cotisations pour chaque adhérent
|
||||||
|
$memberships = $client->getMemberSubscriptions($member->id);
|
||||||
|
|
||||||
|
// on traite les notes (privée, publique, lien ect) contenues dans dolibarr
|
||||||
|
|
||||||
|
$i++;
|
||||||
|
}
|
||||||
|
|
||||||
|
$progressBar->finish();
|
||||||
|
// Logs
|
||||||
|
|
||||||
|
$this->info('Import finished. ' .$i.' members have been imported.');
|
||||||
|
}
|
||||||
|
}
|
||||||
61
app/Services/DolibarrService.php
Normal file
61
app/Services/DolibarrService.php
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Services;
|
||||||
|
|
||||||
|
use Illuminate\Http\Client\ConnectionException;
|
||||||
|
use Illuminate\Http\Client\PendingRequest;
|
||||||
|
use Illuminate\Support\Facades\Http;
|
||||||
|
|
||||||
|
class DolibarrService
|
||||||
|
{
|
||||||
|
protected string $baseUrl;
|
||||||
|
protected string $username;
|
||||||
|
protected string $password;
|
||||||
|
protected string $apiKey;
|
||||||
|
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
$this->baseUrl = config('services.dolibarr.base_url');
|
||||||
|
$this->username = config('services.dolibarr.username');
|
||||||
|
$this->password = config('services.dolibarr.password');
|
||||||
|
$this->apiKey = config('services.dolibarr.api_key');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create an authenticated HTTP client for Dolibarr
|
||||||
|
*/
|
||||||
|
protected function client(): PendingRequest
|
||||||
|
{
|
||||||
|
return Http::withBasicAuth($this->username, $this->password)
|
||||||
|
->withHeaders([
|
||||||
|
'Accept' => 'application/json',
|
||||||
|
'DOLAPIKEY' => $this->apiKey,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get all members
|
||||||
|
* @throws ConnectionException
|
||||||
|
*/
|
||||||
|
public function getAllMembers(int $limit = 400, string $sortField = 't.rowid', string $sortOrder = 'ASC'): array
|
||||||
|
{
|
||||||
|
$response = $this->client()->get($this->baseUrl . '/members', [
|
||||||
|
'sortfield' => $sortField,
|
||||||
|
'sortorder' => $sortOrder,
|
||||||
|
'limit' => $limit,
|
||||||
|
]);
|
||||||
|
|
||||||
|
return $response->json();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get member subscriptions
|
||||||
|
* @throws ConnectionException
|
||||||
|
*/
|
||||||
|
public function getMemberSubscriptions(int|string $id): array
|
||||||
|
{
|
||||||
|
$response = $this->client()->get($this->baseUrl . '/members/'. $id . '/subscriptions');
|
||||||
|
|
||||||
|
return $response->json();
|
||||||
|
}
|
||||||
|
}
|
||||||
89
app/Services/ISPConfigService.php
Normal file
89
app/Services/ISPConfigService.php
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Services;
|
||||||
|
|
||||||
|
use Illuminate\Support\Facades\Log;
|
||||||
|
use SoapClient;
|
||||||
|
use SoapFault;
|
||||||
|
use Exception;
|
||||||
|
|
||||||
|
class ISPConfigService
|
||||||
|
{
|
||||||
|
protected ?SoapClient $client = null;
|
||||||
|
protected ?string $sessionId = null;
|
||||||
|
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ISPConfig Login
|
||||||
|
*/
|
||||||
|
public function connect(string $type): void
|
||||||
|
{
|
||||||
|
// Type = 'hosting' or 'mailbox'
|
||||||
|
$username = $username ?? config('services.ispconfig'.$type.'.username');
|
||||||
|
$password = $password ?? config('services.ispconfig'.$type.'.password');
|
||||||
|
|
||||||
|
try {
|
||||||
|
$this->client = new SoapClient(null, [
|
||||||
|
'location' => config('services.ispconfig' . $type . '.base_url'),
|
||||||
|
'trace' => true,
|
||||||
|
'exceptions' => true,
|
||||||
|
'stream_context' => stream_context_create([
|
||||||
|
'ssl' => [
|
||||||
|
'verify_peer' => false,
|
||||||
|
'verify_peer_name' => false,
|
||||||
|
'allow_self_signed' => true,
|
||||||
|
],
|
||||||
|
]),
|
||||||
|
]);
|
||||||
|
|
||||||
|
$this->sessionId = $this->client->login($username, $password);
|
||||||
|
} catch (SoapFault $e) {
|
||||||
|
throw new Exception("An error occurred : " . $e->getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get all clients
|
||||||
|
*/
|
||||||
|
public function getAllClients(string $username = null, string $password = null): array
|
||||||
|
{
|
||||||
|
if (!$this->sessionId) {
|
||||||
|
$this->connect($username, $password);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$clientIds = $this->client->client_get_all($this->sessionId);
|
||||||
|
$clients = [];
|
||||||
|
|
||||||
|
foreach ($clientIds as $id) {
|
||||||
|
$details = $this->client->client_get($this->sessionId, (int)$id);
|
||||||
|
if (!empty($details)) {
|
||||||
|
$clients[] = (array) $details;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $clients;
|
||||||
|
} catch (SoapFault $e) {
|
||||||
|
throw new Exception("An error occurred : " . $e->getMessage());
|
||||||
|
} finally {
|
||||||
|
$this->disconnect();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logout
|
||||||
|
*/
|
||||||
|
public function disconnect(): void
|
||||||
|
{
|
||||||
|
if ($this->client && $this->sessionId) {
|
||||||
|
try {
|
||||||
|
$this->client->logout($this->sessionId);
|
||||||
|
} catch (SoapFault $e) {
|
||||||
|
Log::info('ISP Config logout succeeded');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -15,7 +15,8 @@
|
|||||||
"laravel/framework": "^12.0",
|
"laravel/framework": "^12.0",
|
||||||
"laravel/tinker": "^2.10",
|
"laravel/tinker": "^2.10",
|
||||||
"laravel/wayfinder": "^0.1.9",
|
"laravel/wayfinder": "^0.1.9",
|
||||||
"spatie/laravel-permission": "^6.21"
|
"spatie/laravel-permission": "^6.21",
|
||||||
|
"ext-soap": "*"
|
||||||
},
|
},
|
||||||
"require-dev": {
|
"require-dev": {
|
||||||
"barryvdh/laravel-debugbar": "^3.16",
|
"barryvdh/laravel-debugbar": "^3.16",
|
||||||
|
|||||||
@@ -35,4 +35,24 @@ return [
|
|||||||
],
|
],
|
||||||
],
|
],
|
||||||
|
|
||||||
|
'dolibarr' => [
|
||||||
|
'base_url' => env('DOLIBARR_URL'),
|
||||||
|
'username' => env('DOLIBARR_USERNAME'),
|
||||||
|
'password' => env('DOLIBARR_PWD'),
|
||||||
|
'api_key' => env('DOLIBARR_APIKEY')
|
||||||
|
],
|
||||||
|
|
||||||
|
'ispconfig' => [
|
||||||
|
'hosting' => [
|
||||||
|
'base_url' => env('HOSTING_ISPAPI_URL'),
|
||||||
|
'username' => env('HOSTING_ISPAPI_USERNAME'),
|
||||||
|
'password' => env('HOSTING_ISPAPI_PWD'),
|
||||||
|
],
|
||||||
|
'mailbox' => [
|
||||||
|
'base_url' => env('MAIL_ISPAPI_URL'),
|
||||||
|
'username' => env('MAIL_ISPAPI_USERNAME'),
|
||||||
|
'password' => env('MAIL_ISPAPI_PWD'),
|
||||||
|
]
|
||||||
|
]
|
||||||
|
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -526,3 +526,77 @@ membershipForm.head = (options?: RouteQueryOptions): RouteFormDefinition<'get'>
|
|||||||
})
|
})
|
||||||
|
|
||||||
membership.form = membershipForm
|
membership.form = membershipForm
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see routes/dev-routes.php:5
|
||||||
|
* @route '/call-dolibarr'
|
||||||
|
*/
|
||||||
|
export const callDolibarr = (options?: RouteQueryOptions): RouteDefinition<'get'> => ({
|
||||||
|
url: callDolibarr.url(options),
|
||||||
|
method: 'get',
|
||||||
|
})
|
||||||
|
|
||||||
|
callDolibarr.definition = {
|
||||||
|
methods: ["get","head"],
|
||||||
|
url: '/call-dolibarr',
|
||||||
|
} satisfies RouteDefinition<["get","head"]>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see routes/dev-routes.php:5
|
||||||
|
* @route '/call-dolibarr'
|
||||||
|
*/
|
||||||
|
callDolibarr.url = (options?: RouteQueryOptions) => {
|
||||||
|
return callDolibarr.definition.url + queryParams(options)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see routes/dev-routes.php:5
|
||||||
|
* @route '/call-dolibarr'
|
||||||
|
*/
|
||||||
|
callDolibarr.get = (options?: RouteQueryOptions): RouteDefinition<'get'> => ({
|
||||||
|
url: callDolibarr.url(options),
|
||||||
|
method: 'get',
|
||||||
|
})
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see routes/dev-routes.php:5
|
||||||
|
* @route '/call-dolibarr'
|
||||||
|
*/
|
||||||
|
callDolibarr.head = (options?: RouteQueryOptions): RouteDefinition<'head'> => ({
|
||||||
|
url: callDolibarr.url(options),
|
||||||
|
method: 'head',
|
||||||
|
})
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see routes/dev-routes.php:5
|
||||||
|
* @route '/call-dolibarr'
|
||||||
|
*/
|
||||||
|
const callDolibarrForm = (options?: RouteQueryOptions): RouteFormDefinition<'get'> => ({
|
||||||
|
action: callDolibarr.url(options),
|
||||||
|
method: 'get',
|
||||||
|
})
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see routes/dev-routes.php:5
|
||||||
|
* @route '/call-dolibarr'
|
||||||
|
*/
|
||||||
|
callDolibarrForm.get = (options?: RouteQueryOptions): RouteFormDefinition<'get'> => ({
|
||||||
|
action: callDolibarr.url(options),
|
||||||
|
method: 'get',
|
||||||
|
})
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see routes/dev-routes.php:5
|
||||||
|
* @route '/call-dolibarr'
|
||||||
|
*/
|
||||||
|
callDolibarrForm.head = (options?: RouteQueryOptions): RouteFormDefinition<'get'> => ({
|
||||||
|
action: callDolibarr.url({
|
||||||
|
[options?.mergeQuery ? 'mergeQuery' : 'query']: {
|
||||||
|
_method: 'HEAD',
|
||||||
|
...(options?.query ?? options?.mergeQuery ?? {}),
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
method: 'get',
|
||||||
|
})
|
||||||
|
|
||||||
|
callDolibarr.form = callDolibarrForm
|
||||||
|
|||||||
13
routes/dev-routes.php
Normal file
13
routes/dev-routes.php
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
use Illuminate\Support\Facades\Route;
|
||||||
|
|
||||||
|
Route::get('/call-dolibarr', function () {
|
||||||
|
$call = new App\Services\DolibarrService;
|
||||||
|
$members = $call->getAllMembers();
|
||||||
|
// find specific
|
||||||
|
$userData = collect($members)->firstWhere('id', 139);
|
||||||
|
|
||||||
|
dd($userData);
|
||||||
|
|
||||||
|
})->name('call-dolibarr');
|
||||||
@@ -17,3 +17,4 @@ Route::middleware(['auth', 'verified'])->group(function () {
|
|||||||
require __DIR__.'/settings.php';
|
require __DIR__.'/settings.php';
|
||||||
require __DIR__.'/auth.php';
|
require __DIR__.'/auth.php';
|
||||||
require __DIR__.'/forms.php';
|
require __DIR__.'/forms.php';
|
||||||
|
require __DIR__.'/dev-routes.php';
|
||||||
|
|||||||
Reference in New Issue
Block a user