feat(cms): sync site assets, revalidate webhook, and document download naming

This commit is contained in:
2026-02-10 23:38:31 +08:00
parent c4969cd4d2
commit b6e18a83ec
27 changed files with 1019 additions and 26 deletions

View File

@@ -0,0 +1,55 @@
<?php
namespace Tests\Feature\Cms;
use App\Models\User;
use Database\Seeders\FinancialWorkflowPermissionsSeeder;
use Illuminate\Foundation\Testing\RefreshDatabase;
use Spatie\Permission\Models\Role;
use Tests\TestCase;
class AdminCmsAccessTest extends TestCase
{
use RefreshDatabase;
protected function setUp(): void
{
parent::setUp();
$this->artisan('db:seed', ['--class' => FinancialWorkflowPermissionsSeeder::class]);
}
public function test_user_without_permissions_cannot_access_admin_articles(): void
{
$user = User::factory()->create();
$this->actingAs($user)
->get('/admin/articles')
->assertForbidden();
}
public function test_secretary_general_can_access_admin_articles_and_create_draft(): void
{
$role = Role::where('name', 'secretary_general')->firstOrFail();
$user = User::factory()->create();
$user->assignRole($role);
$this->actingAs($user)
->get('/admin/articles')
->assertOk();
$res = $this->actingAs($user)->post('/admin/articles', [
'title' => 'Test Article',
'content' => 'Hello world',
'content_type' => 'blog',
'access_level' => 'public',
'save_action' => 'draft',
]);
$res->assertRedirect();
$this->assertDatabaseHas('articles', [
'title' => 'Test Article',
'status' => 'draft',
]);
}
}

View File

@@ -0,0 +1,91 @@
<?php
namespace Tests\Feature\Cms;
use App\Models\Article;
use App\Models\ArticleCategory;
use App\Models\ArticleTag;
use Illuminate\Foundation\Testing\RefreshDatabase;
use Tests\TestCase;
class ArticleApiTest extends TestCase
{
use RefreshDatabase;
public function test_articles_index_only_returns_active_and_public_for_guests(): void
{
Article::factory()->create(['title' => 'Visible']);
Article::factory()->draft()->create(['title' => 'Draft']);
Article::factory()->expired()->create(['title' => 'Expired']);
Article::factory()->scheduled()->create(['title' => 'Scheduled']);
$res = $this->getJson('/api/v1/articles?per_page=50');
$res->assertOk();
$res->assertJsonCount(1, 'data');
$res->assertJsonFragment(['title' => 'Visible']);
$res->assertJsonMissing(['title' => 'Draft']);
$res->assertJsonMissing(['title' => 'Expired']);
$res->assertJsonMissing(['title' => 'Scheduled']);
}
public function test_articles_index_filters_by_type_category_tag_and_search(): void
{
$cat = ArticleCategory::factory()->create(['slug' => 'news']);
$tag = ArticleTag::factory()->create(['slug' => 'health']);
$a1 = Article::factory()->create([
'content_type' => Article::CONTENT_TYPE_NOTICE,
'title' => 'Needle Drop',
'content' => 'alpha beta',
]);
$a1->categories()->attach($cat);
$a1->tags()->attach($tag);
Article::factory()->create([
'content_type' => Article::CONTENT_TYPE_BLOG,
'title' => 'Other',
'content' => 'gamma delta',
]);
$this->getJson('/api/v1/articles?type=notice&per_page=50')
->assertOk()
->assertJsonCount(1, 'data')
->assertJsonFragment(['slug' => $a1->slug]);
$this->getJson('/api/v1/articles?category=news&per_page=50')
->assertOk()
->assertJsonCount(1, 'data')
->assertJsonFragment(['slug' => $a1->slug]);
$this->getJson('/api/v1/articles?tag=health&per_page=50')
->assertOk()
->assertJsonCount(1, 'data')
->assertJsonFragment(['slug' => $a1->slug]);
$this->getJson('/api/v1/articles?search=needle&per_page=50')
->assertOk()
->assertJsonCount(1, 'data')
->assertJsonFragment(['slug' => $a1->slug]);
}
public function test_articles_show_returns_wrapped_data_and_increments_view_count(): void
{
$article = Article::factory()->create([
'view_count' => 0,
'title' => 'My Article',
]);
$this->getJson('/api/v1/articles/'.$article->slug)
->assertOk()
->assertJsonPath('data.slug', $article->slug)
->assertJsonStructure([
'data' => ['id', 'title', 'slug', 'content'],
'related' => [
'*' => ['id', 'title', 'slug', 'content_type'],
],
]);
$this->assertSame(1, $article->refresh()->view_count);
}
}

View File

@@ -0,0 +1,51 @@
<?php
namespace Tests\Feature\Cms;
use App\Models\Article;
use App\Models\Page;
use Illuminate\Foundation\Testing\RefreshDatabase;
use Tests\TestCase;
class HomepageApiTest extends TestCase
{
use RefreshDatabase;
public function test_homepage_endpoint_returns_expected_sections(): void
{
Article::factory()->pinned(0)->create(['content_type' => Article::CONTENT_TYPE_BLOG]);
Article::factory()->pinned(1)->create(['content_type' => Article::CONTENT_TYPE_NOTICE]);
Article::factory()->create(['content_type' => Article::CONTENT_TYPE_BLOG]);
Article::factory()->create(['content_type' => Article::CONTENT_TYPE_NOTICE]);
Article::factory()->create(['content_type' => Article::CONTENT_TYPE_DOCUMENT]);
Article::factory()->create(['content_type' => Article::CONTENT_TYPE_RELATED_NEWS]);
$about = Page::factory()->create(['slug' => 'about']);
$this->getJson('/api/v1/homepage')
->assertOk()
->assertJsonStructure([
'featured' => [
'*' => ['id', 'title', 'slug', 'content_type'],
],
'latest_blog' => [
'*' => ['id', 'title', 'slug', 'content_type'],
],
'latest_notice' => [
'*' => ['id', 'title', 'slug', 'content_type'],
],
'latest_document' => [
'*' => ['id', 'title', 'slug', 'content_type'],
],
'latest_related_news' => [
'*' => ['id', 'title', 'slug', 'content_type'],
],
'about' => ['id', 'slug', 'content'],
'categories' => [
'*' => ['id', 'name', 'slug'],
],
])
->assertJsonPath('about.slug', $about->slug);
}
}

View File

@@ -239,4 +239,44 @@ class DocumentTest extends TestCase
$document->refresh();
$this->assertEquals(2, $document->version_count);
}
/**
* Test public download returns UTF-8 compatible content disposition
*/
public function test_public_document_download_uses_utf8_content_disposition(): void
{
$category = DocumentCategory::factory()->create();
$document = Document::factory()->create([
'created_by_user_id' => $this->admin->id,
'document_category_id' => $category->id,
'access_level' => 'public',
'status' => 'active',
'version_count' => 0,
]);
$filePath = 'documents/charter-v2.pdf';
Storage::disk('private')->put($filePath, 'pdf-content');
Storage::put($filePath, 'pdf-content');
$document->addVersion(
filePath: $filePath,
originalFilename: '台灣尤塞氏症暨視聽弱協會章程V2.pdf',
mimeType: 'application/pdf',
fileSize: Storage::disk('private')->size($filePath),
uploadedBy: $this->admin,
versionNotes: '初始版本'
);
$response = $this->get(route('documents.public.download', [
'uuid' => $document->public_uuid,
]));
$response->assertOk();
$disposition = (string) $response->headers->get('Content-Disposition');
$this->assertStringContainsString(
"filename*=UTF-8''".rawurlencode('台灣尤塞氏症暨視聽弱協會章程V2.pdf'),
$disposition
);
}
}