Files
SkinbaseNova/database/migrations/2026_04_04_000001_create_groups_tables.php

96 lines
3.8 KiB
PHP

<?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('groups', function (Blueprint $table): void {
$table->id();
$table->foreignId('owner_user_id')
->constrained('users')
->cascadeOnDelete();
$table->string('name', 80);
$table->string('slug', 90)->unique();
$table->string('headline', 160)->nullable();
$table->text('bio')->nullable();
$table->enum('visibility', ['public', 'private'])->default('public');
$table->string('website_url', 2048)->nullable();
$table->json('links_json')->nullable();
$table->string('avatar_path')->nullable();
$table->string('banner_path')->nullable();
$table->unsignedInteger('artworks_count')->default(0);
$table->unsignedInteger('collections_count')->default(0);
$table->unsignedInteger('followers_count')->default(0);
$table->timestamp('last_activity_at')->nullable();
$table->timestamps();
$table->index(['visibility', 'followers_count']);
$table->index(['owner_user_id', 'created_at']);
});
Schema::create('group_members', function (Blueprint $table): void {
$table->id();
$table->foreignId('group_id')
->constrained('groups')
->cascadeOnDelete();
$table->foreignId('user_id')
->constrained('users')
->cascadeOnDelete();
$table->foreignId('invited_by_user_id')
->nullable()
->constrained('users')
->nullOnDelete();
$table->enum('role', ['owner', 'admin', 'editor', 'member'])->default('member');
$table->enum('status', ['pending', 'active', 'revoked'])->default('pending');
$table->text('note')->nullable();
$table->timestamp('invited_at')->nullable();
$table->timestamp('expires_at')->nullable();
$table->timestamp('accepted_at')->nullable();
$table->timestamp('revoked_at')->nullable();
$table->timestamps();
$table->unique(['group_id', 'user_id'], 'group_members_group_user_unique');
$table->index(['group_id', 'status', 'role'], 'group_members_status_role_idx');
});
Schema::create('group_follows', function (Blueprint $table): void {
$table->id();
$table->foreignId('group_id')
->constrained('groups')
->cascadeOnDelete();
$table->foreignId('user_id')
->constrained('users')
->cascadeOnDelete();
$table->timestamps();
$table->unique(['group_id', 'user_id'], 'group_follows_group_user_unique');
});
Schema::create('artwork_contributors', function (Blueprint $table): void {
$table->id();
$table->foreignId('artwork_id')
->constrained('artworks')
->cascadeOnDelete();
$table->foreignId('user_id')
->constrained('users')
->cascadeOnDelete();
$table->unsignedInteger('sort_order')->default(0);
$table->timestamps();
$table->unique(['artwork_id', 'user_id'], 'artwork_contributors_artwork_user_unique');
$table->index(['artwork_id', 'sort_order'], 'artwork_contributors_sort_idx');
});
}
public function down(): void
{
Schema::dropIfExists('artwork_contributors');
Schema::dropIfExists('group_follows');
Schema::dropIfExists('group_members');
Schema::dropIfExists('groups');
}
};