diff --git a/app/Http/Controllers/Admin/SiteSubscriptionController.php b/app/Http/Controllers/Admin/SiteSubscriptionController.php
index 157a53d..79a7aa9 100644
--- a/app/Http/Controllers/Admin/SiteSubscriptionController.php
+++ b/app/Http/Controllers/Admin/SiteSubscriptionController.php
@@ -384,6 +384,43 @@ class SiteSubscriptionController extends Controller
return redirect()->back()->with('success', '订阅状态已更新:' . ($this->statusLabels()[$subscription->status] ?? $subscription->status));
}
+ public function batchMarkExpired(Request $request): \Illuminate\Http\RedirectResponse
+ {
+ $this->ensurePlatformAdmin($request);
+
+ // 仅支持在“已过期(expiry=expired)集合”上执行,避免误把正常订阅批量标记为已过期。
+ $filters = [
+ 'status' => trim((string) $request->input('status', '')),
+ 'keyword' => trim((string) $request->input('keyword', '')),
+ 'merchant_id' => trim((string) $request->input('merchant_id', '')),
+ 'plan_id' => trim((string) $request->input('plan_id', '')),
+ 'expiry' => trim((string) $request->input('expiry', '')),
+ ];
+
+ if ((string) ($filters['expiry'] ?? '') !== 'expired') {
+ return redirect()->back()->with('warning', '为避免误操作:批量标记已过期仅允许在「已过期(expiry=expired)」集合视图下执行。');
+ }
+
+ // 防误操作:需要二次确认
+ if ((string) $request->input('confirm', '') !== 'YES') {
+ return redirect()->back()->with('warning', '为避免误操作,请在确认框输入 YES 后再批量标记已过期。');
+ }
+
+ $query = $this->applyFilters(SiteSubscription::query(), $filters);
+
+ // 再加一道硬条件:ends_at 必须 < now(与 expiry=expired 一致)
+ $query->whereNotNull('ends_at')->where('ends_at', '<', now());
+
+ // 仅把“非已过期”的订阅更新为 expired
+ $affected = (clone $query)
+ ->where('status', '!=', 'expired')
+ ->update([
+ 'status' => 'expired',
+ ]);
+
+ return redirect()->back()->with('success', '已批量标记已过期:' . (int) $affected . ' 条。');
+ }
+
protected function applyFilters(Builder $query, array $filters): Builder
{
return $query
diff --git a/resources/views/admin/site_subscriptions/index.blade.php b/resources/views/admin/site_subscriptions/index.blade.php
index ce4a14e..d78ea45 100644
--- a/resources/views/admin/site_subscriptions/index.blade.php
+++ b/resources/views/admin/site_subscriptions/index.blade.php
@@ -237,6 +237,12 @@
工具
+
+ @php
+ $batchMarkExpiredEnabled = (string) ($filters['expiry'] ?? '') === 'expired';
+ $batchMarkExpiredReason = $batchMarkExpiredEnabled ? '' : '请先进入「已过期(expiry=expired)」集合后再执行批量标记。';
+ @endphp
+
@if($attachOrderId > 0)
diff --git a/routes/web.php b/routes/web.php
index 9143165..7b651ae 100644
--- a/routes/web.php
+++ b/routes/web.php
@@ -138,6 +138,7 @@ Route::prefix('admin')->group(function () {
Route::get('/site-subscriptions/export', [SiteSubscriptionController::class, 'export']);
Route::get('/site-subscriptions/{subscription}', [SiteSubscriptionController::class, 'show']);
Route::post('/site-subscriptions/{subscription}/set-status', [SiteSubscriptionController::class, 'setStatus']);
+ Route::post('/site-subscriptions/batch-mark-expired', [SiteSubscriptionController::class, 'batchMarkExpired']);
Route::get('/plans', [PlanController::class, 'index']);
Route::get('/plans/export', [PlanController::class, 'export']);
diff --git a/tests/Feature/AdminSiteSubscriptionSyncFailedReasonLongDoesNotRenderKeywordLinkTest.php b/tests/Feature/AdminSiteSubscriptionSyncFailedReasonLongDoesNotRenderKeywordLinkTest.php
index 80c8e6b..ab65858 100644
--- a/tests/Feature/AdminSiteSubscriptionSyncFailedReasonLongDoesNotRenderKeywordLinkTest.php
+++ b/tests/Feature/AdminSiteSubscriptionSyncFailedReasonLongDoesNotRenderKeywordLinkTest.php
@@ -54,7 +54,8 @@ class AdminSiteSubscriptionSyncFailedReasonLongDoesNotRenderKeywordLinkTest exte
'activated_at' => now()->subDay(),
]);
- $longReason = str_repeat('错', 120);
+ // 超过 config('saasshop.platform_orders.sync_error_keyword_link_max_len', 200)
+ $longReason = str_repeat('错', 220);
PlatformOrder::query()->create([
'merchant_id' => $merchant->id,
diff --git a/tests/Feature/AdminSiteSubscriptionsBatchMarkExpiredShouldBlockWhenNotExpiredScopeTest.php b/tests/Feature/AdminSiteSubscriptionsBatchMarkExpiredShouldBlockWhenNotExpiredScopeTest.php
new file mode 100644
index 0000000..c81d8c2
--- /dev/null
+++ b/tests/Feature/AdminSiteSubscriptionsBatchMarkExpiredShouldBlockWhenNotExpiredScopeTest.php
@@ -0,0 +1,104 @@
+seed();
+
+ $this->post('/admin/login', [
+ 'email' => 'platform.admin@demo.local',
+ 'password' => 'Platform@123456',
+ ])->assertRedirect('/admin');
+ }
+
+ public function test_batch_mark_expired_should_block_when_expiry_not_expired(): void
+ {
+ $this->loginAsPlatformAdmin();
+
+ $merchant = Merchant::query()->firstOrFail();
+ $plan = Plan::query()->create([
+ 'code' => 'sub_batch_mark_expired_block_scope_plan',
+ 'name' => '订阅批量标记过期阻断(scope)测试套餐',
+ 'billing_cycle' => 'monthly',
+ 'price' => 1,
+ 'list_price' => 1,
+ 'status' => 'active',
+ 'sort' => 10,
+ 'published_at' => now(),
+ ]);
+
+ $sub = SiteSubscription::query()->create([
+ 'merchant_id' => $merchant->id,
+ 'plan_id' => $plan->id,
+ 'status' => 'activated',
+ 'source' => 'manual',
+ 'subscription_no' => 'SUB_BATCH_EXPIRED_BLOCK_SCOPE_0001',
+ 'plan_name' => $plan->name,
+ 'billing_cycle' => $plan->billing_cycle,
+ 'period_months' => 1,
+ 'amount' => 1,
+ 'starts_at' => now()->subDay(),
+ 'ends_at' => now()->addDays(5),
+ 'activated_at' => now()->subDay(),
+ ]);
+
+ $this->post('/admin/site-subscriptions/batch-mark-expired', [
+ 'expiry' => 'expiring_7d',
+ 'confirm' => 'YES',
+ ])->assertRedirect()->assertSessionHas('warning');
+
+ $sub->refresh();
+ $this->assertSame('activated', (string) $sub->status);
+ }
+
+ public function test_batch_mark_expired_should_block_when_confirm_missing(): void
+ {
+ $this->loginAsPlatformAdmin();
+
+ $merchant = Merchant::query()->firstOrFail();
+ $plan = Plan::query()->create([
+ 'code' => 'sub_batch_mark_expired_block_confirm_plan',
+ 'name' => '订阅批量标记过期阻断(confirm)测试套餐',
+ 'billing_cycle' => 'monthly',
+ 'price' => 1,
+ 'list_price' => 1,
+ 'status' => 'active',
+ 'sort' => 10,
+ 'published_at' => now(),
+ ]);
+
+ $sub = SiteSubscription::query()->create([
+ 'merchant_id' => $merchant->id,
+ 'plan_id' => $plan->id,
+ 'status' => 'activated',
+ 'source' => 'manual',
+ 'subscription_no' => 'SUB_BATCH_EXPIRED_BLOCK_CONFIRM_0001',
+ 'plan_name' => $plan->name,
+ 'billing_cycle' => $plan->billing_cycle,
+ 'period_months' => 1,
+ 'amount' => 1,
+ 'starts_at' => now()->subDay(),
+ 'ends_at' => now()->subDays(1),
+ 'activated_at' => now()->subDay(),
+ ]);
+
+ $this->post('/admin/site-subscriptions/batch-mark-expired', [
+ 'expiry' => 'expired',
+ 'confirm' => '',
+ ])->assertRedirect()->assertSessionHas('warning');
+
+ $sub->refresh();
+ $this->assertSame('activated', (string) $sub->status);
+ }
+}
diff --git a/tests/Feature/AdminSiteSubscriptionsBatchMarkExpiredShouldMarkExpiredOnlyWhenEndsAtPastTest.php b/tests/Feature/AdminSiteSubscriptionsBatchMarkExpiredShouldMarkExpiredOnlyWhenEndsAtPastTest.php
new file mode 100644
index 0000000..1d8248f
--- /dev/null
+++ b/tests/Feature/AdminSiteSubscriptionsBatchMarkExpiredShouldMarkExpiredOnlyWhenEndsAtPastTest.php
@@ -0,0 +1,99 @@
+seed();
+
+ $this->post('/admin/login', [
+ 'email' => 'platform.admin@demo.local',
+ 'password' => 'Platform@123456',
+ ])->assertRedirect('/admin');
+ }
+
+ public function test_batch_mark_expired_should_only_update_subscriptions_with_ends_at_past(): void
+ {
+ $this->loginAsPlatformAdmin();
+
+ $merchant = Merchant::query()->firstOrFail();
+ $plan = Plan::query()->create([
+ 'code' => 'sub_batch_mark_expired_success_plan',
+ 'name' => '订阅批量标记过期(成功)测试套餐',
+ 'billing_cycle' => 'monthly',
+ 'price' => 1,
+ 'list_price' => 1,
+ 'status' => 'active',
+ 'sort' => 10,
+ 'published_at' => now(),
+ ]);
+
+ $expiredA = SiteSubscription::query()->create([
+ 'merchant_id' => $merchant->id,
+ 'plan_id' => $plan->id,
+ 'status' => 'activated',
+ 'source' => 'manual',
+ 'subscription_no' => 'SUB_BATCH_EXPIRED_OK_0001',
+ 'plan_name' => $plan->name,
+ 'billing_cycle' => $plan->billing_cycle,
+ 'period_months' => 1,
+ 'amount' => 1,
+ 'starts_at' => now()->subDays(40),
+ 'ends_at' => now()->subDays(1),
+ 'activated_at' => now()->subDays(40),
+ ]);
+
+ $expiredAlready = SiteSubscription::query()->create([
+ 'merchant_id' => $merchant->id,
+ 'plan_id' => $plan->id,
+ 'status' => 'expired',
+ 'source' => 'manual',
+ 'subscription_no' => 'SUB_BATCH_EXPIRED_ALREADY_0001',
+ 'plan_name' => $plan->name,
+ 'billing_cycle' => $plan->billing_cycle,
+ 'period_months' => 1,
+ 'amount' => 1,
+ 'starts_at' => now()->subDays(70),
+ 'ends_at' => now()->subDays(10),
+ 'activated_at' => now()->subDays(70),
+ ]);
+
+ $notExpired = SiteSubscription::query()->create([
+ 'merchant_id' => $merchant->id,
+ 'plan_id' => $plan->id,
+ 'status' => 'activated',
+ 'source' => 'manual',
+ 'subscription_no' => 'SUB_BATCH_EXPIRED_SKIP_0001',
+ 'plan_name' => $plan->name,
+ 'billing_cycle' => $plan->billing_cycle,
+ 'period_months' => 1,
+ 'amount' => 1,
+ 'starts_at' => now()->subDays(5),
+ 'ends_at' => now()->addDays(5),
+ 'activated_at' => now()->subDays(5),
+ ]);
+
+ $this->post('/admin/site-subscriptions/batch-mark-expired', [
+ 'expiry' => 'expired',
+ 'confirm' => 'YES',
+ ])->assertRedirect()->assertSessionHas('success');
+
+ $expiredA->refresh();
+ $expiredAlready->refresh();
+ $notExpired->refresh();
+
+ $this->assertSame('expired', (string) $expiredA->status);
+ $this->assertSame('expired', (string) $expiredAlready->status);
+ $this->assertSame('activated', (string) $notExpired->status);
+ }
+}