TaskController.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493
  1. <?php
  2. namespace App\Http\Controllers\API;
  3. use App\Exports\DownloadHelper;
  4. use App\Exports\TaskExport;
  5. use App\Http\Controllers\Controller;
  6. use App\Http\Requests\API\Task\AssignRequest;
  7. use App\Http\Requests\API\Task\BatchCreateItemRules;
  8. use App\Http\Requests\API\Task\BatchCreateRequest;
  9. use App\Http\Requests\API\Task\CreateOrUpdateRequest;
  10. use App\Http\Requests\API\Task\LinkContainerRequest;
  11. use App\Http\Resources\API\TaskDetailResource;
  12. use App\Http\Resources\API\TaskResource;
  13. use App\Imports\TaskImport;
  14. use App\Models\Container;
  15. use App\Models\Enums\ActionObjectType;
  16. use App\Models\Enums\FileObjectType;
  17. use App\Models\Enums\ObjectAction;
  18. use App\Models\Enums\TaskStatus;
  19. use App\Models\Library;
  20. use App\Models\Project;
  21. use App\Models\Requirement;
  22. use App\Models\Task;
  23. use App\Models\TaskContainer;
  24. use App\Repositories\ActionRepository;
  25. use App\Repositories\CustomFieldRepository;
  26. use App\Services\File\FileAssociationService;
  27. use App\Services\History\ModelChangeDetector;
  28. use Carbon\Carbon;
  29. use Illuminate\Http\Request;
  30. use Illuminate\Support\Facades\Auth;
  31. use Illuminate\Support\Facades\Validator;
  32. use Maatwebsite\Excel\Facades\Excel;
  33. class TaskController extends Controller
  34. {
  35. use DownloadHelper;
  36. /**
  37. * Display a listing of the resource.
  38. */
  39. public function index(Request $request)
  40. {
  41. $pageSize=$request->get('page_size') ?? 10;
  42. $sort=$request->input('sort','desc');
  43. $tasks = Task::query()
  44. ->where("parent_id", 0)
  45. ->with(['children', 'assignTo', 'createdBy'])
  46. ->filter($request->all())
  47. ->allowed()
  48. ->orderBy('created_at',$sort)
  49. ->paginate($pageSize);
  50. return TaskResource::collection($tasks);
  51. }
  52. public function export(Request $request)
  53. {
  54. return $this->downloadExcelHelper(
  55. new TaskExport(),
  56. "task",
  57. $request->get("extension"),
  58. );
  59. }
  60. /**
  61. * Store a newly created resource in storage.
  62. */
  63. public function store(FileAssociationService $service, CreateOrUpdateRequest $request, CustomFieldRepository $customFieldRepo)
  64. {
  65. $isAction=true;
  66. $projectId=$request->get("project_id");
  67. $project = Project::allowed($projectId,$isAction)->find($projectId);
  68. if($project == null){
  69. return $this->badRequest('Permission denied or project not found. Please contact the administrator.');
  70. };
  71. $requirement = $request->has('requirement_id')
  72. ? Requirement::query()->findOrFail($request->get("requirement_id"))
  73. : null;
  74. $service->check(
  75. $request->get("file_ids", []),
  76. FileObjectType::TASK,
  77. $request->get("file_uuid"),
  78. );
  79. $formData = [
  80. ...$request->all(),
  81. 'company_id' => Auth::user()->company_id,
  82. 'created_by' => Auth::id(),
  83. 'whitelist' => $request->whitelist ? sprintf(",%s,", implode(',', $request->whitelist)) : null,
  84. 'asset_id' => $requirement?->asset_id,
  85. 'description' => $request->description? (new \App\Services\File\ImageUrlService)->interceptImageUrl($request->description) : null,
  86. 'requirement_group_id'=> $requirement?->requirement_group_id,
  87. ];
  88. if ($request->has("naming_rule_id") && $request->get("naming_rule_id") > 0) {
  89. $keys = $customFieldRepo->keysByGroup($request->get("naming_rule_id"));
  90. $formData['naming_rules'] = $request->only($keys);
  91. }
  92. $task = Task::create($formData);
  93. ActionRepository::createByTask($task, ObjectAction::CREATED);
  94. $service->association($task->id);
  95. return $this->created();
  96. }
  97. public function import(Request $request)
  98. {
  99. Excel::import(new TaskImport(), $request->file("file"));
  100. return $this->created();
  101. }
  102. /**
  103. * Display the specified resource.
  104. */
  105. public function show(string $id)
  106. {
  107. $task = Task::query()->allowed($id)->with([
  108. 'containers'
  109. ])->findOrFail($id);
  110. return new TaskDetailResource($task);
  111. }
  112. public function start(Request $request, string $id)
  113. {
  114. $isAction=true;
  115. $task = Task::query()->allowed($id,$isAction)->find($id);
  116. if($task==null){
  117. return $this->badRequest('Permission denied or task not found. Please contact the administrator.');
  118. };
  119. $task->status = TaskStatus::DOING->value;
  120. $changes = ModelChangeDetector::detector(ActionObjectType::TASK, $task);
  121. $task->save();
  122. ActionRepository::createByTask(
  123. $task,
  124. ObjectAction::STARTED,
  125. $request->comment?(new \App\Services\File\ImageUrlService)->interceptImageUrl($request->comment) : null,
  126. objectChanges: $changes
  127. );
  128. return $this->noContent();
  129. }
  130. public function pause(Request $request, string $id)
  131. {
  132. $isAction=true;
  133. $task = Task::query()->allowed($id,$isAction)->find($id);
  134. if($task==null){
  135. return $this->badRequest('Permission denied or task not found. Please contact the administrator.');
  136. };
  137. $task->status = TaskStatus::PAUSE->value;
  138. $changes = ModelChangeDetector::detector(ActionObjectType::TASK, $task);
  139. $task->save();
  140. ActionRepository::createByTask(
  141. $task, ObjectAction::PAUSED,
  142. $request->comment?(new \App\Services\File\ImageUrlService)->interceptImageUrl($request->comment) : null,
  143. objectChanges: $changes
  144. );
  145. return $this->noContent();
  146. }
  147. public function closed(Request $request, string $id)
  148. {
  149. $isAction=true;
  150. $task = Task::query()->allowed($id,$isAction)->find($id);
  151. if($task==null){
  152. return $this->badRequest('Permission denied or task not found. Please contact the administrator.');
  153. };
  154. $task->status = TaskStatus::CLOSED->value;
  155. $changes = ModelChangeDetector::detector(ActionObjectType::TASK, $task);
  156. $task->save();
  157. ActionRepository::createByTask(
  158. $task, ObjectAction::CLOSED,
  159. $request->comment?(new \App\Services\File\ImageUrlService)->interceptImageUrl($request->comment) : null,
  160. objectChanges: $changes
  161. );
  162. return $this->noContent();
  163. }
  164. public function done(Request $request, string $id)
  165. {
  166. $isAction=true;
  167. $task = Task::query()->allowed($id,$isAction)->find($id);
  168. if($task==null){
  169. return $this->badRequest('Permission denied or task not found. Please contact the administrator.');
  170. };
  171. $task->fill([
  172. 'status' => TaskStatus::DONE->value,
  173. 'finished_by' => Auth::user()->id,
  174. 'finished_at' => Carbon::now(),
  175. ]);
  176. $changes = ModelChangeDetector::detector(ActionObjectType::TASK, $task);
  177. $task->save();
  178. ActionRepository::createByTask(
  179. $task, ObjectAction::DONE,
  180. $request->comment?(new \App\Services\File\ImageUrlService)->interceptImageUrl($request->comment) : null,
  181. objectChanges: $changes
  182. );
  183. return $this->noContent();
  184. }
  185. public function cancel(Request $request, string $id)
  186. {
  187. $isAction=true;
  188. $task = Task::query()->allowed($id,$isAction)->find($id);
  189. if($task==null){
  190. return $this->badRequest('Permission denied or task not found. Please contact the administrator.');
  191. };
  192. $task->fill([
  193. 'status' => TaskStatus::CANCEL->value,
  194. 'canceled_by' => Auth::user()->id,
  195. 'canceled_at' => Carbon::now(),
  196. ]);
  197. $changes = ModelChangeDetector::detector(ActionObjectType::TASK, $task);
  198. $task->save();
  199. ActionRepository::createByTask(
  200. $task, ObjectAction::CANCELED,
  201. $request->comment?(new \App\Services\File\ImageUrlService)->interceptImageUrl($request->comment) : null,
  202. objectChanges: $changes
  203. );
  204. return $this->noContent();
  205. }
  206. public function wait(Request $request,string $id){
  207. $isAction=true;
  208. $task = Task::query()->allowed($id,$isAction)->find($id);
  209. if($task==null){
  210. return $this->badRequest('Permission denied or task not found. Please contact the administrator.');
  211. };
  212. $task->status=TaskStatus::WAIT->value;
  213. $changes = ModelChangeDetector::detector(ActionObjectType::TASK, $task);
  214. $task->save();
  215. ActionRepository::createByTask(
  216. $task, ObjectAction::WAITED,
  217. $request->comment?(new \App\Services\File\ImageUrlService)->interceptImageUrl($request->comment) : null,
  218. objectChanges: $changes
  219. );
  220. return $this->noContent();
  221. }
  222. public function assign(AssignRequest $request,string $id){
  223. $isAction=true;
  224. $task = Task::query()->allowed($id,$isAction)->find($id);
  225. if($task==null){
  226. return $this->badRequest('Permission denied or task not found. Please contact the administrator.');
  227. };
  228. $task->fill([
  229. 'assign'=>$request->get('assign'),
  230. ...$request->all(),
  231. ]);
  232. $changes = ModelChangeDetector::detector(ActionObjectType::TASK, $task);
  233. $task->save();
  234. ActionRepository::createByTask(
  235. $task, ObjectAction::ASSIGNED,
  236. $request->comment?(new \App\Services\File\ImageUrlService)->interceptImageUrl($request->comment) : null,
  237. objectChanges: $changes
  238. );
  239. return $this->noContent();
  240. }
  241. /**
  242. * Update the specified resource in storage.
  243. */
  244. public function update(CreateOrUpdateRequest $request, string $id, CustomFieldRepository $customFieldRepo)
  245. {
  246. $isAction=true;
  247. $task = Task::query()->allowed($id,$isAction)->find($id);
  248. if($task==null){
  249. return $this->badRequest('Permission denied or task not found. Please contact the administrator.');
  250. };
  251. $requirement = $request->has('requirement_id') && $task->requirement_id != $request->get('requirement_id')
  252. ? Requirement::query()->findOrFail($request->get("requirement_id"))
  253. : null;
  254. $formData = [...$request->all(),
  255. 'whitelist' => $request->whitelist ? sprintf(",%s,", implode(',', $request->whitelist)) : null,
  256. 'description' => $request->description? (new \App\Services\File\ImageUrlService)->interceptImageUrl($request->description) : null,
  257. 'asset_id' => $requirement?->asset_id,
  258. ];
  259. if ($request->has("naming_rule_id") && $request->get("naming_rule_id") > 0) {
  260. $keys = $customFieldRepo->keysByGroup($request->get("naming_rule_id"));
  261. $formData['naming_rules'] = $request->only($keys);
  262. }
  263. $task->fill($formData);
  264. $changes = ModelChangeDetector::detector(ActionObjectType::TASK, $task);
  265. $task->save();
  266. ActionRepository::createByTask($task, ObjectAction::EDITED, objectChanges: $changes);
  267. return $this->noContent();
  268. }
  269. /**
  270. * Remove the specified resource from storage.
  271. */
  272. public function destroy(string $id)
  273. {
  274. $isAction=true;
  275. $task = Task::query()->allowed($id,$isAction)->find($id);
  276. if($task==null){
  277. return $this->badRequest('Permission denied or task not found. Please contact the administrator.');
  278. };
  279. $task->delete();
  280. ActionRepository::createByTask($task, ObjectAction::DELETED);
  281. return $this->noContent();
  282. }
  283. public function batchStore(BatchCreateRequest $request, CustomFieldRepository $customFieldRepo)
  284. {
  285. $isAction=true;
  286. $projectId=$request->get("project_id");
  287. $project = Project::allowed($projectId,$isAction)->find($projectId);
  288. if($project==null){
  289. return $this->badRequest('Permission denied or project not found. Please contact the administrator.');
  290. };
  291. $parsedItems = [];
  292. $previousItem = [];
  293. foreach ($request->items as $index => $item) {
  294. if ($index == 0) {
  295. $newItem = $item;
  296. } else {
  297. $newItem = [];
  298. foreach ($item as $k => $v) {
  299. $newItem[$k] = $v == "ditto" ? data_get($previousItem, $k) : $v;
  300. }
  301. }
  302. $previousItem = $newItem;
  303. $parsedItems[] = $newItem;
  304. }
  305. $itemRules = new BatchCreateItemRules();
  306. foreach ($parsedItems as $index => $item) {
  307. $rules = $itemRules->rules($item);
  308. try {
  309. $validator = Validator::make($item, $rules);
  310. if ($validator->fails()) {
  311. return $this->badRequest(sprintf("line: %d, %s", $index + 1, $validator->errors()->first()));
  312. }
  313. } catch (\Exception $e) {
  314. return $this->badRequest(sprintf("line: %d, %s", $index + 1, $e->getMessage()));
  315. }
  316. }
  317. foreach ($parsedItems as $item) {
  318. $requirement=$item['requirement_id']>0 ? Requirement::query()->findOrFail($item['requirement_id']) : null;
  319. $item["whitelist"]=!empty($item['whitelist']) ? sprintf(",%s,", implode(',',$item['whitelist'])) : null;
  320. $namingRuleId = data_get($item, "naming_rule_id", 0);
  321. if ($namingRuleId > 0) {
  322. $keys = $customFieldRepo->keysByGroup($namingRuleId);
  323. $item['naming_rules'] = collect($item)->only($keys)->toArray();
  324. }
  325. $task = Task::query()->create([
  326. ...$item,
  327. 'project_id' => $project->id,
  328. 'parent_id' => $request->parent_id,
  329. 'company_id' => Auth::user()->company_id,
  330. 'created_by' => Auth::id(),
  331. 'asset_id' => $requirement?->asset_id,
  332. 'requirement_group_id'=> $requirement?->requirement_group_id,
  333. ]);
  334. ActionRepository::createByTask($task, ObjectAction::CREATED);
  335. }
  336. return $this->noContent();
  337. }
  338. /**
  339. * 容器链接
  340. *
  341. * @param LinkContainerRequest $request
  342. * @param string $id
  343. * @return \Illuminate\Http\JsonResponse|\Illuminate\Http\Response
  344. */
  345. public function linkContainer(LinkContainerRequest $request, string $id)
  346. {
  347. $isAction=true;
  348. $task = Task::query()->allowed($id,$isAction)->find($id);
  349. if($task==null){
  350. return $this->badRequest('Permission denied or task not found. Please contact the administrator.');
  351. };
  352. $libraryIds = Library::query()->allowed()->where("project_id", $task->project_id)->pluck("id");
  353. if ($libraryIds->isEmpty()) {
  354. return $this->badRequest("No container exists for the current task item");
  355. }
  356. $containers = Container::query()
  357. ->allowed()
  358. ->whereIn("library_id", $libraryIds->toArray())
  359. ->whereIn("id", $request->get("container_ids", []))
  360. ->get(['id']);
  361. if (! $containers) {
  362. return $this->badRequest("Please select a valid container");
  363. }
  364. foreach ($containers as $container) {
  365. TaskContainer::query()->firstOrCreate([
  366. 'container_id' => $container->id,
  367. 'task_id' => $task->id
  368. ]);
  369. }
  370. return $this->noContent();
  371. }
  372. public function unlinkContainer(string $id)
  373. {
  374. $taskContainer = TaskContainer::query()->findOrFail($id);
  375. $taskId=$taskContainer->task_id;
  376. $isAction=true;
  377. $task = Task::query()->allowed($taskId,$isAction)->find($taskId);
  378. if($task==null){
  379. return $this->badRequest('Permission denied or task not found. Please contact the administrator.');
  380. };
  381. $taskContainer->delete();
  382. return $this->noContent();
  383. }
  384. /**
  385. * 待关联的容器
  386. *
  387. * @param Request $request
  388. * @param string $id
  389. * @return \Illuminate\Http\JsonResponse
  390. */
  391. public function containerToBeLinked(Request $request, string $id)
  392. {
  393. $task = Task::query()->allowed($id)->findOrFail($id);
  394. $libraryIds = Library::query()->allowed()->where("project_id", $task->project_id)->pluck("id");
  395. if ($libraryIds->isEmpty()) {
  396. return $this->badRequest("No container exists for the current task item");
  397. }
  398. $containers = Container::query()
  399. ->allowed()
  400. ->leftJoin("task_container", "containers.id", "=", "task_container.container_id")
  401. ->where($request->only(['library_id']))
  402. ->whereIn("library_id", $libraryIds->toArray())
  403. ->whereNull("task_container.id")
  404. ->selectRaw("containers.id, containers.name")
  405. ->get();
  406. return $this->success([
  407. 'data' => $containers
  408. ]);
  409. }
  410. }