ScheduleControllerTest.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472
  1. <?php
  2. namespace Tests\Feature;
  3. use App\Models\Dictionary\Area;
  4. use App\Models\Dictionary\District;
  5. use App\Models\MafOrder;
  6. use App\Models\Order;
  7. use App\Models\Permission;
  8. use App\Models\Product;
  9. use App\Models\ProductSKU;
  10. use App\Models\Reclamation;
  11. use App\Models\Role;
  12. use App\Models\Schedule;
  13. use App\Models\User;
  14. use Illuminate\Foundation\Testing\RefreshDatabase;
  15. use Illuminate\Support\Facades\Bus;
  16. use Tests\TestCase;
  17. class ScheduleControllerTest extends TestCase
  18. {
  19. use RefreshDatabase;
  20. protected $seed = true;
  21. private User $adminUser;
  22. private User $managerUser;
  23. private User $brigadierUser;
  24. protected function setUp(): void
  25. {
  26. parent::setUp();
  27. $this->adminUser = User::factory()->create(['role' => Role::ADMIN]);
  28. $this->managerUser = User::factory()->create(['role' => Role::MANAGER]);
  29. $this->brigadierUser = User::factory()->create(['role' => Role::BRIGADIER]);
  30. }
  31. // ==================== Authentication ====================
  32. public function test_guest_cannot_access_schedule_index(): void
  33. {
  34. $response = $this->get(route('schedule.index'));
  35. $response->assertRedirect(route('login'));
  36. }
  37. public function test_guest_cannot_create_schedule_from_order(): void
  38. {
  39. $response = $this->post(route('schedule.create-from-order'), []);
  40. $response->assertRedirect(route('login'));
  41. }
  42. public function test_guest_cannot_update_schedule(): void
  43. {
  44. $response = $this->post(route('schedule.update'), []);
  45. $response->assertRedirect(route('login'));
  46. }
  47. public function test_guest_cannot_delete_schedule(): void
  48. {
  49. $brigadier = User::factory()->create(['role' => Role::BRIGADIER]);
  50. $schedule = Schedule::factory()->create(['brigadier_id' => $brigadier->id]);
  51. $response = $this->delete(route('schedule.delete', $schedule));
  52. $response->assertRedirect(route('login'));
  53. }
  54. // ==================== Authorization ====================
  55. public function test_manager_cannot_create_schedule_from_order(): void
  56. {
  57. $response = $this->actingAs($this->managerUser)
  58. ->post(route('schedule.create-from-order'), []);
  59. $response->assertStatus(403);
  60. }
  61. public function test_manager_cannot_update_schedule(): void
  62. {
  63. $response = $this->actingAs($this->managerUser)
  64. ->post(route('schedule.update'), []);
  65. $response->assertStatus(403);
  66. }
  67. public function test_manager_cannot_delete_schedule(): void
  68. {
  69. $brigadier = User::factory()->create(['role' => Role::BRIGADIER]);
  70. $schedule = Schedule::factory()->create(['brigadier_id' => $brigadier->id]);
  71. $response = $this->actingAs($this->managerUser)
  72. ->delete(route('schedule.delete', $schedule));
  73. $response->assertStatus(403);
  74. }
  75. // ==================== Index ====================
  76. public function test_admin_can_access_schedule_index(): void
  77. {
  78. $response = $this->actingAs($this->adminUser)
  79. ->get(route('schedule.index'));
  80. $response->assertStatus(200);
  81. $response->assertViewIs('schedule.index');
  82. }
  83. public function test_brigadier_can_access_schedule_index(): void
  84. {
  85. $response = $this->actingAs($this->brigadierUser)
  86. ->get(route('schedule.index'));
  87. $response->assertStatus(200);
  88. }
  89. public function test_brigadier_sees_only_visible_platform_and_reclamation_schedules(): void
  90. {
  91. $otherBrigadier = User::factory()->create(['role' => Role::BRIGADIER]);
  92. $visibleOrder = Order::factory()->create([
  93. 'brigadier_id' => $this->brigadierUser->id,
  94. 'order_status_id' => Order::STATUS_IN_MOUNT,
  95. 'object_address' => 'ул. Площадка видимая, д. 1',
  96. ]);
  97. $hiddenOrder = Order::factory()->create([
  98. 'brigadier_id' => $this->brigadierUser->id,
  99. 'order_status_id' => Order::STATUS_HANDED_OVER,
  100. 'object_address' => 'ул. Площадка скрытая, д. 2',
  101. ]);
  102. $visibleReclamation = Reclamation::factory()->create([
  103. 'order_id' => $visibleOrder->id,
  104. 'brigadier_id' => $this->brigadierUser->id,
  105. 'status_id' => Reclamation::STATUS_IN_WORK,
  106. 'reason' => 'Рекламация видимая',
  107. ]);
  108. $hiddenReclamation = Reclamation::factory()->create([
  109. 'order_id' => $hiddenOrder->id,
  110. 'brigadier_id' => $this->brigadierUser->id,
  111. 'status_id' => Reclamation::STATUS_DONE,
  112. 'reason' => 'Рекламация скрытая',
  113. ]);
  114. $foreignOrder = Order::factory()->create([
  115. 'brigadier_id' => $otherBrigadier->id,
  116. 'order_status_id' => Order::STATUS_IN_MOUNT,
  117. 'object_address' => 'ул. Чужая площадка, д. 3',
  118. ]);
  119. $foreignReclamation = Reclamation::factory()->create([
  120. 'order_id' => $foreignOrder->id,
  121. 'brigadier_id' => $otherBrigadier->id,
  122. 'status_id' => Reclamation::STATUS_IN_WORK,
  123. 'reason' => 'Чужая рекламация',
  124. ]);
  125. Schedule::factory()->create([
  126. 'source' => 'Площадки',
  127. 'order_id' => $visibleOrder->id,
  128. 'brigadier_id' => $this->brigadierUser->id,
  129. 'installation_date' => '2026-03-16',
  130. 'object_address' => $visibleOrder->object_address,
  131. ]);
  132. Schedule::factory()->create([
  133. 'source' => 'Площадки',
  134. 'order_id' => $hiddenOrder->id,
  135. 'brigadier_id' => $this->brigadierUser->id,
  136. 'installation_date' => '2026-03-16',
  137. 'object_address' => $hiddenOrder->object_address,
  138. ]);
  139. Schedule::factory()->create([
  140. 'source' => 'Рекламации',
  141. 'address_code' => 'РЕКЛ-' . $visibleReclamation->id,
  142. 'order_id' => $visibleOrder->id,
  143. 'brigadier_id' => $this->brigadierUser->id,
  144. 'installation_date' => '2026-03-17',
  145. 'object_address' => $visibleOrder->object_address,
  146. 'object_type' => $visibleReclamation->reason,
  147. ]);
  148. Schedule::factory()->create([
  149. 'source' => 'Рекламации',
  150. 'address_code' => 'РЕКЛ-' . $hiddenReclamation->id,
  151. 'order_id' => $hiddenOrder->id,
  152. 'brigadier_id' => $this->brigadierUser->id,
  153. 'installation_date' => '2026-03-18',
  154. 'object_address' => $hiddenOrder->object_address,
  155. 'object_type' => $hiddenReclamation->reason,
  156. ]);
  157. Schedule::factory()->create([
  158. 'source' => 'Площадки',
  159. 'order_id' => $foreignOrder->id,
  160. 'brigadier_id' => $otherBrigadier->id,
  161. 'installation_date' => '2026-03-19',
  162. 'object_address' => $foreignOrder->object_address,
  163. ]);
  164. Schedule::factory()->create([
  165. 'source' => 'Рекламации',
  166. 'address_code' => 'РЕКЛ-' . $foreignReclamation->id,
  167. 'order_id' => $foreignOrder->id,
  168. 'brigadier_id' => $otherBrigadier->id,
  169. 'installation_date' => '2026-03-20',
  170. 'object_address' => $foreignOrder->object_address,
  171. 'object_type' => $foreignReclamation->reason,
  172. ]);
  173. $response = $this->actingAs($this->brigadierUser)
  174. ->get(route('schedule.index', ['year' => 2026, 'week' => 12]));
  175. $response->assertStatus(200);
  176. $response->assertSee($visibleOrder->object_address);
  177. $response->assertSee($visibleReclamation->reason);
  178. $response->assertDontSee($hiddenOrder->object_address);
  179. $response->assertDontSee($hiddenReclamation->reason);
  180. $response->assertDontSee($foreignOrder->object_address);
  181. $response->assertDontSee($foreignReclamation->reason);
  182. }
  183. public function test_custom_role_with_brigadier_schedule_scope_uses_brigadier_visibility(): void
  184. {
  185. $permissionIds = Permission::query()
  186. ->whereIn('slug', ['schedule.view', 'schedule.scope.brigadier'])
  187. ->pluck('id')
  188. ->mapWithKeys(fn (int $id): array => [$id => ['effect' => 'allow']]);
  189. $role = Role::query()->create([
  190. 'slug' => 'custom_brigadier_scope',
  191. 'name' => 'Custom brigadier scope',
  192. 'is_system' => false,
  193. 'is_active' => true,
  194. ]);
  195. $role->permissions()->sync($permissionIds);
  196. $customUser = User::factory()->create([
  197. 'role' => $role->slug,
  198. 'role_id' => $role->id,
  199. ]);
  200. $otherUser = User::factory()->create(['role' => Role::BRIGADIER]);
  201. $visibleOrder = Order::factory()->create([
  202. 'brigadier_id' => $customUser->id,
  203. 'order_status_id' => Order::STATUS_IN_MOUNT,
  204. 'object_address' => 'ул. Кастомная видимая, д. 1',
  205. ]);
  206. $foreignOrder = Order::factory()->create([
  207. 'brigadier_id' => $otherUser->id,
  208. 'order_status_id' => Order::STATUS_IN_MOUNT,
  209. 'object_address' => 'ул. Кастомная чужая, д. 2',
  210. ]);
  211. Schedule::factory()->create([
  212. 'source' => 'Площадки',
  213. 'order_id' => $visibleOrder->id,
  214. 'brigadier_id' => $customUser->id,
  215. 'installation_date' => '2026-03-16',
  216. 'object_address' => $visibleOrder->object_address,
  217. ]);
  218. Schedule::factory()->create([
  219. 'source' => 'Площадки',
  220. 'order_id' => $foreignOrder->id,
  221. 'brigadier_id' => $otherUser->id,
  222. 'installation_date' => '2026-03-16',
  223. 'object_address' => $foreignOrder->object_address,
  224. ]);
  225. $response = $this->actingAs($customUser)
  226. ->get(route('schedule.index', ['year' => 2026, 'week' => 12]));
  227. $response->assertOk();
  228. $response->assertSee($visibleOrder->object_address);
  229. $response->assertDontSee($foreignOrder->object_address);
  230. }
  231. // ==================== Update (create manual schedule) ====================
  232. public function test_admin_can_create_manual_schedule(): void
  233. {
  234. Bus::fake();
  235. $district = District::query()->first();
  236. $area = Area::query()->first();
  237. $brigadier = User::factory()->create(['role' => Role::BRIGADIER]);
  238. $response = $this->actingAs($this->adminUser)
  239. ->post(route('schedule.update'), [
  240. 'installation_date' => '2026-03-15',
  241. 'address_code' => 'TEST-001',
  242. 'object_address' => 'ул. Тестовая, 1',
  243. 'object_type' => 'Площадка',
  244. 'mafs' => 'МАФ-001 - 2',
  245. 'mafs_count' => 2,
  246. 'district_id' => $district->id,
  247. 'area_id' => $area->id,
  248. 'brigadier_id' => $brigadier->id,
  249. 'comment' => 'Тестовый комментарий',
  250. ]);
  251. $response->assertRedirect();
  252. $this->assertDatabaseHas('schedules', [
  253. 'address_code' => 'TEST-001',
  254. 'object_address' => 'ул. Тестовая, 1',
  255. 'manual' => true,
  256. ]);
  257. }
  258. public function test_admin_can_update_existing_schedule(): void
  259. {
  260. Bus::fake();
  261. $district = District::query()->first();
  262. $area = Area::query()->first();
  263. $brigadier = User::factory()->create(['role' => Role::BRIGADIER]);
  264. $schedule = Schedule::factory()->create([
  265. 'installation_date' => '2026-03-10',
  266. 'object_address' => 'Старый адрес',
  267. 'brigadier_id' => $brigadier->id,
  268. ]);
  269. $response = $this->actingAs($this->adminUser)
  270. ->post(route('schedule.update'), [
  271. 'id' => $schedule->id,
  272. 'installation_date' => '2026-03-20',
  273. 'address_code' => $schedule->address_code ?? 'UPD-001',
  274. 'object_address' => 'Новый адрес',
  275. 'object_type' => 'Площадка',
  276. 'mafs' => 'МАФ-002 - 1',
  277. 'mafs_count' => 1,
  278. 'district_id' => $district->id,
  279. 'area_id' => $area->id,
  280. 'brigadier_id' => $brigadier->id,
  281. 'comment' => '',
  282. ]);
  283. $response->assertRedirect();
  284. $this->assertDatabaseHas('schedules', [
  285. 'id' => $schedule->id,
  286. 'object_address' => 'Новый адрес',
  287. 'installation_date' => '2026-03-20',
  288. ]);
  289. }
  290. public function test_cannot_create_schedule_from_order_when_order_has_unlinked_maf(): void
  291. {
  292. $product = Product::factory()->create();
  293. $order = Order::factory()->readyToMount()->withBrigadier($this->brigadierUser)->create([
  294. 'installation_date' => '2026-04-22',
  295. 'install_days' => 2,
  296. ]);
  297. ProductSKU::factory()->create([
  298. 'order_id' => $order->id,
  299. 'product_id' => $product->id,
  300. 'maf_order_id' => null,
  301. ]);
  302. $response = $this->actingAs($this->adminUser)
  303. ->post(route('schedule.create-from-order'), [
  304. 'order_id' => $order->id,
  305. 'delete_old_records' => '1',
  306. ]);
  307. $response->assertRedirect(route('order.show', $order->id));
  308. $response->assertSessionHas('danger', function ($messages) {
  309. return in_array('МАФ не привязан к заказу', (array) $messages, true);
  310. });
  311. $this->assertDatabaseCount('schedules', 0);
  312. }
  313. public function test_cannot_create_schedule_from_order_when_maf_order_number_is_empty(): void
  314. {
  315. $product = Product::factory()->create();
  316. $order = Order::factory()->readyToMount()->withBrigadier($this->brigadierUser)->create([
  317. 'installation_date' => '2026-04-22',
  318. 'install_days' => 2,
  319. ]);
  320. $mafOrder = MafOrder::factory()->create([
  321. 'product_id' => $product->id,
  322. 'order_number' => '',
  323. ]);
  324. ProductSKU::factory()->create([
  325. 'order_id' => $order->id,
  326. 'product_id' => $product->id,
  327. 'maf_order_id' => $mafOrder->id,
  328. ]);
  329. Schedule::factory()->create([
  330. 'order_id' => $order->id,
  331. 'source' => 'Площадки',
  332. 'manual' => false,
  333. 'brigadier_id' => $this->brigadierUser->id,
  334. 'object_address' => $order->object_address,
  335. 'installation_date' => '2026-04-20',
  336. ]);
  337. $response = $this->actingAs($this->adminUser)
  338. ->post(route('schedule.create-from-order'), [
  339. 'order_id' => $order->id,
  340. 'delete_old_records' => '1',
  341. ]);
  342. $response->assertRedirect(route('order.show', $order->id));
  343. $response->assertSessionHas('danger', function ($messages) {
  344. return in_array('МАФ не привязан к заказу', (array) $messages, true);
  345. });
  346. $this->assertDatabaseCount('schedules', 1);
  347. }
  348. public function test_can_create_schedule_from_order_when_all_mafs_are_linked_to_order_numbers(): void
  349. {
  350. $product = Product::factory()->create();
  351. $order = Order::factory()->readyToMount()->withBrigadier($this->brigadierUser)->create([
  352. 'installation_date' => '2026-04-22',
  353. 'install_days' => 2,
  354. ]);
  355. $mafOrder = MafOrder::factory()->create([
  356. 'product_id' => $product->id,
  357. 'order_number' => 'MO-2026-1',
  358. ]);
  359. ProductSKU::factory()->create([
  360. 'order_id' => $order->id,
  361. 'product_id' => $product->id,
  362. 'maf_order_id' => $mafOrder->id,
  363. ]);
  364. $response = $this->actingAs($this->adminUser)
  365. ->post(route('schedule.create-from-order'), [
  366. 'order_id' => $order->id,
  367. 'comment' => '',
  368. ]);
  369. $response->assertRedirect(route('schedule.index'));
  370. $this->assertDatabaseCount('schedules', 2);
  371. $this->assertDatabaseHas('schedules', [
  372. 'order_id' => $order->id,
  373. 'source' => 'Площадки',
  374. 'manual' => false,
  375. ]);
  376. }
  377. // ==================== Delete ====================
  378. public function test_admin_can_delete_schedule(): void
  379. {
  380. $brigadier = User::factory()->create(['role' => Role::BRIGADIER]);
  381. $schedule = Schedule::factory()->create(['brigadier_id' => $brigadier->id]);
  382. $response = $this->actingAs($this->adminUser)
  383. ->delete(route('schedule.delete', $schedule));
  384. $response->assertRedirect();
  385. $this->assertDatabaseMissing('schedules', ['id' => $schedule->id]);
  386. }
  387. // ==================== Export ====================
  388. public function test_admin_can_export_schedule(): void
  389. {
  390. Bus::fake();
  391. $response = $this->actingAs($this->adminUser)
  392. ->post(route('schedule.export'), [
  393. 'start_date' => '2026-03-01',
  394. 'end_date' => '2026-03-31',
  395. 'week' => '10',
  396. 'year' => 2026,
  397. ]);
  398. $response->assertRedirect(route('schedule.index', ['week' => 10, 'year' => 2026]));
  399. $response->assertSessionHas('success');
  400. }
  401. }