Update pricing UX, billing flows, and API rules

This commit is contained in:
Dwindi Ramadhana
2026-02-12 00:52:40 +07:00
parent cf065fab1e
commit a905256353
202 changed files with 22348 additions and 301 deletions

View File

@@ -0,0 +1,22 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
public function up(): void
{
Schema::table('users', function (Blueprint $table) {
$table->string('tier', 20)->default('free')->after('password');
});
}
public function down(): void
{
Schema::table('users', function (Blueprint $table) {
$table->dropColumn('tier');
});
}
};

View File

@@ -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
{
public function up(): void
{
Schema::create('user_api_keys', function (Blueprint $table) {
$table->id();
$table->foreignId('user_id')->constrained()->cascadeOnDelete();
$table->string('key_hash', 64)->unique();
$table->string('key_prefix', 12)->index();
$table->string('name', 100)->nullable();
$table->timestamp('last_used_at')->nullable();
$table->timestamp('revoked_at')->nullable();
$table->timestamps();
});
}
public function down(): void
{
Schema::dropIfExists('user_api_keys');
}
};

View File

@@ -0,0 +1,28 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
public function up(): void
{
Schema::create('user_keywords', function (Blueprint $table) {
$table->id();
$table->foreignId('user_id')->constrained()->cascadeOnDelete();
$table->string('emoji_slug');
$table->string('keyword', 200);
$table->string('lang', 10)->default('und');
$table->timestamps();
$table->unique(['user_id', 'emoji_slug', 'keyword']);
$table->index(['user_id', 'emoji_slug']);
});
}
public function down(): void
{
Schema::dropIfExists('user_keywords');
}
};

View File

@@ -0,0 +1,32 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
public function up(): void
{
Schema::create('subscriptions', function (Blueprint $table) {
$table->id();
$table->foreignId('user_id')->constrained()->cascadeOnDelete();
$table->string('plan', 20);
$table->string('status', 20)->default('active');
$table->string('provider', 20)->nullable();
$table->string('provider_ref', 100)->nullable();
$table->timestamp('started_at')->nullable();
$table->timestamp('expires_at')->nullable();
$table->timestamp('canceled_at')->nullable();
$table->timestamp('next_renewal_at')->nullable();
$table->timestamps();
$table->index(['user_id', 'status']);
});
}
public function down(): void
{
Schema::dropIfExists('subscriptions');
}
};

View File

@@ -0,0 +1,28 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
public function up(): void
{
Schema::create('pricing_plans', function (Blueprint $table) {
$table->id();
$table->string('code', 30)->unique();
$table->string('name', 50);
$table->string('currency', 10)->default('IDR');
$table->unsignedBigInteger('amount');
$table->string('period', 20)->nullable();
$table->string('status', 20)->default('active');
$table->json('meta')->nullable();
$table->timestamps();
});
}
public function down(): void
{
Schema::dropIfExists('pricing_plans');
}
};

View File

@@ -0,0 +1,24 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
public function up(): void
{
Schema::create('pricing_changes', function (Blueprint $table) {
$table->id();
$table->string('admin_ref', 120)->nullable();
$table->json('before')->nullable();
$table->json('after')->nullable();
$table->timestamps();
});
}
public function down(): void
{
Schema::dropIfExists('pricing_changes');
}
};

View File

@@ -0,0 +1,24 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
public function up(): void
{
Schema::create('settings', function (Blueprint $table): void {
$table->id();
$table->string('key', 120)->unique();
$table->json('value')->nullable();
$table->string('updated_by', 120)->nullable();
$table->timestamps();
});
}
public function down(): void
{
Schema::dropIfExists('settings');
}
};

View File

@@ -0,0 +1,28 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
public function up(): void
{
Schema::create('webhook_events', function (Blueprint $table): void {
$table->id();
$table->string('provider', 50);
$table->string('event_type', 120)->nullable();
$table->string('status', 50)->default('received');
$table->json('payload')->nullable();
$table->text('error')->nullable();
$table->timestamp('received_at')->nullable();
$table->timestamp('processed_at')->nullable();
$table->timestamps();
});
}
public function down(): void
{
Schema::dropIfExists('webhook_events');
}
};

View File

@@ -0,0 +1,26 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
public function up(): void
{
Schema::table('webhook_events', function (Blueprint $table): void {
$table->string('event_id', 120)->nullable()->after('provider');
$table->json('headers')->nullable()->after('payload');
$table->index(['provider', 'event_id']);
});
}
public function down(): void
{
Schema::table('webhook_events', function (Blueprint $table): void {
$table->dropIndex(['provider', 'event_id']);
$table->dropColumn('event_id');
$table->dropColumn('headers');
});
}
};

View File

@@ -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::table('users', function (Blueprint $table) {
$table->string('role', 32)->default('user')->after('password');
$table->index('role');
});
}
/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::table('users', function (Blueprint $table) {
$table->dropIndex(['role']);
$table->dropColumn('role');
});
}
};

View File

@@ -0,0 +1,25 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration {
public function up(): void
{
Schema::create('admin_audit_logs', function (Blueprint $table): void {
$table->id();
$table->unsignedBigInteger('admin_id')->nullable()->index();
$table->string('admin_email', 255)->nullable()->index();
$table->string('action', 64)->index();
$table->json('payload')->nullable();
$table->string('ip_address', 64)->nullable();
$table->timestamps();
});
}
public function down(): void
{
Schema::dropIfExists('admin_audit_logs');
}
};

View File

@@ -0,0 +1,32 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
public function up(): void
{
Schema::create('orders', function (Blueprint $table): void {
$table->id();
$table->foreignId('user_id')->constrained()->cascadeOnDelete();
$table->string('plan_code', 40);
$table->string('type', 20); // one_time | subscription
$table->string('currency', 10);
$table->unsignedInteger('amount');
$table->string('status', 20)->default('pending');
$table->string('provider', 20)->nullable();
$table->string('provider_ref', 100)->nullable();
$table->timestamps();
$table->index(['user_id', 'status']);
$table->index(['provider', 'provider_ref']);
});
}
public function down(): void
{
Schema::dropIfExists('orders');
}
};

View File

@@ -0,0 +1,34 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
public function up(): void
{
Schema::create('payments', function (Blueprint $table): void {
$table->id();
$table->foreignId('user_id')->constrained()->cascadeOnDelete();
$table->foreignId('order_id')->constrained()->cascadeOnDelete();
$table->string('provider', 20);
$table->string('type', 20); // one_time | subscription
$table->string('plan_code', 40);
$table->string('currency', 10);
$table->unsignedInteger('amount');
$table->string('status', 20)->default('pending');
$table->string('provider_ref', 100)->nullable();
$table->json('raw_payload')->nullable();
$table->timestamps();
$table->index(['user_id', 'status']);
$table->index(['provider', 'provider_ref']);
});
}
public function down(): void
{
Schema::dropIfExists('payments');
}
};

View File

@@ -15,11 +15,6 @@ class DatabaseSeeder extends Seeder
*/
public function run(): void
{
// User::factory(10)->create();
User::factory()->create([
'name' => 'Test User',
'email' => 'test@example.com',
]);
$this->call(PricingPlanSeeder::class);
}
}

View File

@@ -0,0 +1,27 @@
<?php
namespace Database\Seeders;
use App\Models\PricingPlan;
use Illuminate\Database\Seeder;
class PricingPlanSeeder extends Seeder
{
public function run(): void
{
$defaults = config('dewemoji.pricing.defaults', []);
foreach ($defaults as $plan) {
PricingPlan::updateOrCreate(
['code' => $plan['code']],
[
'name' => $plan['name'],
'currency' => $plan['currency'],
'amount' => $plan['amount'],
'period' => $plan['period'],
'status' => $plan['status'],
'meta' => $plan['meta'] ?? null,
]
);
}
}
}