Skip to content
48 changes: 48 additions & 0 deletions ProcessMaker/Http/Controllers/Api/DevLinkController.php
Original file line number Diff line number Diff line change
Expand Up @@ -2,17 +2,23 @@

namespace ProcessMaker\Http\Controllers\Api;

use Illuminate\Database\Eloquent\ModelNotFoundException;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Notification;
use Illuminate\Validation\Rule;
use ProcessMaker\Exception\ValidationException;
use ProcessMaker\Http\Controllers\Controller;
use ProcessMaker\Http\Resources\ApiCollection;
use ProcessMaker\Jobs\DevLinkInstall;
use ProcessMaker\Models\Bundle;
use ProcessMaker\Models\BundleAsset;
use ProcessMaker\Models\BundleInstance;
use ProcessMaker\Models\BundleSetting;
use ProcessMaker\Models\DevLink;
use ProcessMaker\Models\Setting;
use ProcessMaker\Models\SettingsMenus;
use ProcessMaker\Models\User;
use ProcessMaker\Notifications\BundleUpdatedNotification;

class DevLinkController extends Controller
{
Expand Down Expand Up @@ -154,12 +160,34 @@ public function updateBundle(Request $request, Bundle $bundle)

public function increaseBundleVersion(Bundle $bundle)
{
$bundle->notifyBundleUpdated();

$bundle->version = $bundle->version + 1;
$bundle->saveOrFail();

return $bundle;
}

public function bundleUpdated($bundleId, $token)
{
try {
$bundle = Bundle::where('remote_id', $bundleId)->firstOrFail();
} catch (ModelNotFoundException $e) {
return response()->json(['error' => 'Bundle not found'], 403);
}

$storedToken = $bundle->webhook_token;

if ($token !== $storedToken) {
return response()->json(['error' => 'Invalid token'], 403);
}
$adminUsers = User::where('is_administrator', true)->get();

Notification::send($adminUsers, new BundleUpdatedNotification($bundle));

return $bundle;
}

public function deleteBundle(Bundle $bundle)
{
$bundle->delete();
Expand Down Expand Up @@ -198,6 +226,14 @@ public function reinstallBundle(Request $request, Bundle $bundle)
];
}

public function addBundleInstance(Request $request, Bundle $bundle)
{
BundleInstance::create([
'bundle_id' => $bundle->id,
'instance_url' => $request->input('instance_url'),
]);
}

public function exportLocalBundle(Bundle $bundle)
{
return ['payloads' => $bundle->export()];
Expand Down Expand Up @@ -330,4 +366,16 @@ public function deleteBundleSetting(BundleSetting $bundleSetting)

return response()->json(['message' => 'Bundle setting deleted.'], 200);
}

public function getBundleSetting(Bundle $bundle, $settingKey)
{
$setting = $bundle->settings()->where('setting', $settingKey)->first();

return $setting;
}

public function getBundleAllSettings($settingKey)
{
return Setting::where([['group_id', SettingsMenus::getId($settingKey)], ['hidden', 0]])->get();
}
}
2 changes: 2 additions & 0 deletions ProcessMaker/Jobs/DevLinkInstall.php
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,8 @@ public function __construct(
*/
public function handle(): void
{
//log
\Log::info('DevLinkInstall job started: ' . $this->devLinkId);
$devLink = DevLink::findOrFail($this->devLinkId);
$logger = new Logger($this->userId);

Expand Down
29 changes: 29 additions & 0 deletions ProcessMaker/Models/Bundle.php
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
namespace ProcessMaker\Models;

use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Support\Facades\Http;
use ProcessMaker\Exception\ExporterNotSupported;
use ProcessMaker\Exception\ValidationException;
use ProcessMaker\ImportExport\Importer;
Expand Down Expand Up @@ -173,6 +174,16 @@ public function addSettings($setting, $newId, $type = null)
if (json_last_error() === JSON_ERROR_NONE) {
if (is_array($decodedNewId)) {
$newId = $decodedNewId;
if ($existingSetting) {
if ($decodedNewId['id'] === []) {
$existingSetting->delete();

return;
}
$existingSetting->update(['config' => json_encode($newId)]);

return;
}
} elseif (isset($decodedNewId['id'])) {
$newId = [$decodedNewId['id']];
} else {
Expand Down Expand Up @@ -400,4 +411,22 @@ public function reinstall(string $mode, Logger $logger = null)

$logger?->setStatus('done');
}

public function notifyBundleUpdated()
{
$bundleInstances = BundleInstance::where('bundle_id', $this->id)->get();
foreach ($bundleInstances as $bundleInstance) {
$url = $bundleInstance->instance_url;

try {
$response = Http::post($url);

if ($response->status() === 403) {
\Log::error("Failed to notify bundle update for URL: $url " . $response);
}
} catch (\Exception $e) {
\Log::error('Error notifying bundle update: ' . $e->getMessage());
}
}
}
}
21 changes: 21 additions & 0 deletions ProcessMaker/Models/BundleInstance.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
<?php

namespace ProcessMaker\Models;

use Illuminate\Database\Eloquent\Factories\HasFactory;
use ProcessMaker\Models\ProcessMakerModel;

class BundleInstance extends ProcessMakerModel
{
use HasFactory;

protected $fillable = [
'bundle_id',
'instance_url',
];

public function bundle()
{
return $this->belongsTo(Bundle::class);
}
}
9 changes: 9 additions & 0 deletions ProcessMaker/Models/DevLink.php
Original file line number Diff line number Diff line change
Expand Up @@ -138,6 +138,14 @@ public function installRemoteBundle($remoteBundleId, $updateType)
$this->logger->status(__('Downloading bundle from remote instance'));

$bundleInfo = $this->remoteBundle($remoteBundleId)->json();
$token = Str::random(60);

$addBundleInstance = $this->client()->post(
route('api.devlink.add-bundle-instance', ['bundle' => $remoteBundleId], false),
[
'instance_url' => env('APP_URL') . '/devlink/bundle-updated/' . $remoteBundleId . '/' . $token,
]
);

$bundleExport = $this->client()->get(
route('api.devlink.export-local-bundle', ['bundle' => $remoteBundleId], false)
Expand All @@ -160,6 +168,7 @@ public function installRemoteBundle($remoteBundleId, $updateType)
'name' => $bundleInfo['name'],
'published' => $bundleInfo['published'],
'version' => $bundleInfo['version'],
'webhook_token' => $token,
]
);

Expand Down
81 changes: 81 additions & 0 deletions ProcessMaker/Notifications/BundleUpdatedNotification.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
<?php

namespace ProcessMaker\Notifications;

use Illuminate\Bus\Queueable;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Notifications\Messages\BroadcastMessage;
use Illuminate\Notifications\Messages\MailMessage;
use Illuminate\Notifications\Notification;
use ProcessMaker\Models\Bundle;

class BundleUpdatedNotification extends Notification
{
use Queueable;

private $bundleName;

private $bundleUid;

/**
* Create a new notification instance.
*/
public function __construct(Bundle $bundle)
{
$this->bundleName = $bundle->name;
$this->bundleUid = $bundle->id;
}

/**
* Get the notification's delivery channels.
*
* @return array<int, string>
*/
public function via(object $notifiable): array
{
return ['broadcast', NotificationChannel::class];
}

/**
* Get the mail representation of the notification.
*/
public function toMail(object $notifiable): MailMessage
{
return (new MailMessage)
->line('The introduction to the notification.')
->action('Notification Action', url('/'))
->line('Thank you for using our application!');
}

/**
* Get the array representation of the notification.
*
* @return array<string, mixed>
*/
public function toArray($notifiable)
{
$bundle = Bundle::find($this->bundleUid);

return [
'type' => 'BUNDLE_UPDATED',
'message' => sprintf('Bundle updated: %s', $this->bundleName),
'dateTime' => $bundle->updated_at->toIso8601String(),
'name' => $this->bundleName,
'bundleName' => $this->bundleName,
'url' => sprintf(
'/admin/devlink/local-bundles/%s/',
$this->bundleUid
),
];
}

public function toDatabase($notifiable)
{
return $this->toArray($notifiable);
}

public function toBroadcast($notifiable)
{
return new BroadcastMessage($this->toArray($notifiable));
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
<?php

use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;

return new class extends Migration {
/**
* Run the migrations.
*/
public function up(): void
{
Schema::create('bundle_instances', function (Blueprint $table) {
$table->id();
$table->unsignedBigInteger('bundle_id');
$table->string('instance_url');
$table->timestamps();

$table->index(['bundle_id']);
});
}

/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::dropIfExists('bundle_instances');
}
};
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
<?php

use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;

return new class extends Migration {
/**
* Run the migrations.
*/
public function up(): void
{
Schema::table('bundles', function (Blueprint $table) {
$table->string('webhook_token')->nullable();
});
}

/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::table('bundles', function (Blueprint $table) {
$table->dropColumn('webhook_token');
});
}
};
16 changes: 12 additions & 4 deletions resources/js/admin/devlink/components/BundleConfigurations.vue
Original file line number Diff line number Diff line change
Expand Up @@ -10,14 +10,22 @@
<div class="config-status">{{ status(config) }}</div>
</div>
<div class="config-action">
<button class="config-action-button">
<i class="fp-bpmn-data-connector"></i>
<button
class="config-action-button"
@click="$emit('open-settings-modal', {
key: config.type,
value: $event,
settingId: isInSettings(config)?.id,
type: props.type
})"
>
<i class="fp-bpmn-data-connector" />
</button>
<b-form-checkbox
:checked="!!isInSettings(config)"
switch
:disabled="props.disabled"
@change="$emit('config-change', {
@change="$emit('config-change', {
key: config.type,
value: $event,
settingId: isInSettings(config)?.id,
Expand Down Expand Up @@ -60,7 +68,7 @@ const props = defineProps({
}
});

defineEmits(['config-change']);
defineEmits(['config-change', 'open-settings-modal']);

const isInSettings = (type) => {
return (props.values || []).find(value => value.setting === type.type);
Expand Down
Loading