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');
}
};