1
0
mirror of https://github.com/invoiceninja/invoiceninja.git synced 2024-11-14 15:13:29 +01:00
invoiceninja/tests/Feature/TaskApiTest.php

783 lines
20 KiB
PHP
Raw Normal View History

2020-10-12 22:42:02 +02:00
<?php
/**
* Invoice Ninja (https://invoiceninja.com).
*
* @link https://github.com/invoiceninja/invoiceninja source repository
*
* @copyright Copyright (c) 2021. Invoice Ninja LLC (https://invoiceninja.com)
2020-10-12 22:42:02 +02:00
*
* @license https://www.elastic.co/licensing/elastic-license
2020-10-12 22:42:02 +02:00
*/
2020-10-12 22:42:02 +02:00
namespace Tests\Feature;
2022-11-23 00:01:37 +01:00
use App\Models\Task;
2020-10-12 22:42:02 +02:00
use App\Utils\Traits\MakesHash;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Foundation\Testing\DatabaseTransactions;
use Illuminate\Support\Facades\Session;
2021-03-20 01:25:44 +01:00
use Illuminate\Validation\ValidationException;
2020-10-12 22:42:02 +02:00
use Tests\MockAccountData;
use Tests\TestCase;
/**
* @test
* @covers App\Http\Controllers\TaskController
*/
class TaskApiTest extends TestCase
{
use MakesHash;
use DatabaseTransactions;
use MockAccountData;
protected function setUp() :void
2020-10-12 22:42:02 +02:00
{
parent::setUp();
$this->makeTestData();
Session::start();
$this->faker = \Faker\Factory::create();
Model::reguard();
}
2023-02-08 01:27:38 +01:00
private function checkTimeLog(array $log): bool
2023-02-08 00:32:33 +01:00
{
2023-02-16 02:36:09 +01:00
if (count($log) == 0) {
2023-02-08 01:27:38 +01:00
return true;
2023-02-16 02:36:09 +01:00
}
2023-02-08 00:32:33 +01:00
2023-02-08 01:27:38 +01:00
/*Get first value of all arrays*/
2023-02-08 00:32:33 +01:00
$result = array_column($log, 0);
2023-02-08 01:27:38 +01:00
/*Sort the array in ascending order*/
2023-02-08 00:32:33 +01:00
asort($result);
$new_array = [];
2023-02-08 01:27:38 +01:00
/*Rebuild the array in order*/
2023-02-16 02:36:09 +01:00
foreach ($result as $key => $value) {
2023-02-08 00:32:33 +01:00
$new_array[] = $log[$key];
2023-02-16 02:36:09 +01:00
}
2023-02-08 00:32:33 +01:00
2023-02-08 01:27:38 +01:00
/*Iterate through the array and perform checks*/
2023-02-16 02:36:09 +01:00
foreach ($new_array as $key => $array) {
2023-02-08 01:27:38 +01:00
/*Flag which helps us know if there is a NEXT timelog*/
2023-02-08 00:32:33 +01:00
$next = false;
2023-02-08 01:27:38 +01:00
/* If there are more than 1 time log in the array, ensure the last timestamp is not zero*/
2023-02-16 02:36:09 +01:00
if (count($new_array) >1 && $array[1] == 0) {
2023-02-08 00:32:33 +01:00
return false;
2023-02-16 02:36:09 +01:00
}
2023-02-08 00:59:36 +01:00
2023-02-16 02:36:09 +01:00
/* Check if the start time is greater than the end time */
/* Ignore the last value for now, we'll do a separate check for this */
if ($array[0] > $array[1] && $array[1] != 0) {
2023-02-08 00:32:33 +01:00
return false;
2023-02-16 02:36:09 +01:00
}
2023-02-08 01:27:38 +01:00
/* Find the next time log value - if it exists */
2023-02-16 02:36:09 +01:00
if (array_key_exists($key+1, $new_array)) {
2023-02-08 00:32:33 +01:00
$next = $new_array[$key+1];
2023-02-16 02:36:09 +01:00
}
2023-02-08 00:32:33 +01:00
/* check the next time log and ensure the start time is GREATER than the end time of the previous record */
2023-02-16 02:36:09 +01:00
if ($next && $next[0] < $array[1]) {
2023-02-08 00:32:33 +01:00
return false;
2023-02-16 02:36:09 +01:00
}
2023-02-08 00:32:33 +01:00
2023-02-08 01:27:38 +01:00
/* Get the last row of the timelog*/
2023-02-08 00:32:33 +01:00
$last_row = end($new_array);
2023-02-08 01:27:38 +01:00
/*If the last value is NOT zero, ensure start time is not GREATER than the endtime */
2023-02-16 02:36:09 +01:00
if ($last_row[1] != 0 && $last_row[0] > $last_row[1]) {
2023-02-08 00:32:33 +01:00
return false;
2023-02-16 02:36:09 +01:00
}
2023-02-08 00:32:33 +01:00
return true;
}
}
2023-02-08 01:27:38 +01:00
2023-04-11 09:27:37 +02:00
public function testMultiSortArray()
{
$logs = [
[1680035007,1680036807,"",true],
];
$key_values = array_column($logs, 0);
2023-04-12 02:21:02 +02:00
array_multisort($key_values, SORT_ASC, $logs);
$start = $logs[0];
$this->assertEquals(1680035007, $start[0]);
$logs = [
];
$key_values = array_column($logs, 0);
array_multisort($key_values, SORT_ASC, $logs);
$this->assertIsArray($logs);
2023-04-11 09:27:37 +02:00
}
public function testKsortPerformance()
{
$logs = [
[1680035007,1680036807,"",true],
[1681156840,1681158000,"",true],
[1680302433,1680387960,"",true],
[1680715620,1680722820,"",true],
[1,1680737460,"",true]
];
$key_values = array_column($logs, 0);
array_multisort($key_values, SORT_ASC, $logs);
$start = $logs[0];
$this->assertEquals(1, $start[0]);
}
2023-04-11 04:23:09 +02:00
public function testStartStopSanity()
{
$task = Task::factory()->create([
'client_id' => $this->client->id,
'user_id' => $this->user->id,
'company_id' => $this->company->id,
'description' => 'Test Task',
'time_log' => '[[1681165417,1681165432,"sumtin",true],[1681165446,0]]',
]);
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->putJson("/api/v1/tasks/{$task->hashed_id}?stop=true");
$response->assertStatus(200);
}
public function testStoppingTaskWithDescription()
{
$task = Task::factory()->create([
'client_id' => $this->client->id,
'user_id' => $this->user->id,
'company_id' => $this->company->id,
'description' => 'Test Task',
'time_log' => '[[1681165417,1681165432,"sumtin",true],[1681165446,0]]',
]);
$task_repo = new \App\Repositories\TaskRepository();
$task = $task_repo->stop($task);
$log = json_decode($task->time_log);
$last = end($log);
$this->assertNotEquals(0, $last[1]);
$this->assertCount(2, $last);
}
public function testMultiDimensionArrayOfTimes()
{
$logs = [
'[[1680302433,1680387960,"",true]]',
'[[1680715620,1680722820,"",true],[1680729660,1680737460,"",true]]',
'[[1681156840,1681158000,"",true]]',
'[[1680035007,1680036807,"",true]]',
];
foreach($logs as $log)
{
$this->assertTrue($this->checkTimeLog(json_decode($log)));
}
}
public function testArrayOfTimes()
{
$logs = [
"[[1675275148,1675277829]]",
"[[1675375200,1675384200],[1676074247,1676074266]]",
"[[1675443600,1675461600],[1676053305,1676055950],[1676063112,1676067834]]",
"[[1676068200,1676070900]]",
"[[1678134638,1678156238]]",
"[[1678132800,1678134582],[1678134727,1678136801]]",
"[[1678343569,1678344469]]",
"[[1678744339,1678755139]]",
"[[1678894860,1678906620]]",
"[[1679339870,1679341672]]",
"[[1680547478,1680547482]]",
"[[1681156881,0]]",
];
foreach($logs as $log)
{
$this->assertTrue($this->checkTimeLog(json_decode($log)));
}
}
2023-02-08 00:32:33 +01:00
public function testTimeLogChecker1()
{
$log = [
[50,0]
];
$this->assertTrue($this->checkTimeLog($log));
}
public function testTimeLogChecker2()
{
$log = [
[4,5],
[5,1]
];
$this->assertFalse($this->checkTimeLog($log));
}
public function testTimeLogChecker3()
{
$log = [
[4,5],
[3,50]
];
$this->assertFalse($this->checkTimeLog($log));
}
public function testTimeLogChecker4()
{
$log = [
[4,5],
[3,0]
];
$this->assertFalse($this->checkTimeLog($log));
}
public function testTimeLogChecker5()
{
$log = [
[4,5],
[3,1]
];
$this->assertFalse($this->checkTimeLog($log));
}
public function testTimeLogChecker6()
{
$log = [
[4,5],
[1,3],
];
$this->assertTrue($this->checkTimeLog($log));
}
public function testTimeLogChecker7()
{
$log = [
[1,3],
[4,5]
];
$this->assertTrue($this->checkTimeLog($log));
}
2023-02-08 00:59:36 +01:00
public function testTimeLogChecker8()
2023-02-08 00:32:33 +01:00
{
$log = [
[1,3],
[50,0]
];
$this->assertTrue($this->checkTimeLog($log));
}
2022-11-23 00:01:37 +01:00
2023-02-08 00:59:36 +01:00
public function testTimeLogChecker9()
{
$log = [
[4,5,'bb'],
[50,0,'aa'],
];
$this->assertTrue($this->checkTimeLog($log));
}
public function testTimeLogChecker10()
{
$log = [
[4,5,'5'],
[50,0,'3'],
];
$this->assertTrue($this->checkTimeLog($log));
}
public function testTimeLogChecker11()
{
$log = [
[1,2,'a'],
[3,4,'d'],
];
$this->assertTrue($this->checkTimeLog($log));
}
2023-01-19 01:52:07 +01:00
2023-02-16 23:34:50 +01:00
public function testTimeLogChecker12()
{
$log = [
[1,2,'a',true],
[3,4,'d',false],
];
$this->assertTrue($this->checkTimeLog($log));
}
2023-01-19 01:52:07 +01:00
public function testTaskListClientStatus()
{
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->get('/api/v1/tasks?client_status=invoiced')
->assertStatus(200);
}
2022-11-23 00:01:37 +01:00
public function testTaskLockingGate()
{
$data = [
'timelog' => [[1,2,'a'],[3,4,'d']],
2022-11-23 00:01:37 +01:00
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks', $data);
$arr = $response->json();
$response->assertStatus(200);
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->putJson('/api/v1/tasks/' . $arr['data']['id'], $data);
$arr = $response->json();
$response->assertStatus(200);
$task = Task::find($this->decodePrimaryKey($arr['data']['id']));
$task->invoice_id = $this->invoice->id;
$task->save();
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->putJson('/api/v1/tasks/' . $arr['data']['id'], $data);
$arr = $response->json();
$response->assertStatus(200);
$task = Task::find($this->decodePrimaryKey($arr['data']['id']));
2022-11-25 13:30:03 +01:00
$task->company->invoice_task_lock = true;
2022-11-23 00:01:37 +01:00
$task->invoice_id = $this->invoice->id;
2022-11-25 13:30:03 +01:00
$task->push();
2022-11-23 00:01:37 +01:00
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->putJson('/api/v1/tasks/' . $arr['data']['id'], $data);
$arr = $response->json();
$response->assertStatus(401);
}
2022-11-25 13:30:03 +01:00
// public function testTaskLocking()
// {
// $data = [
// 'timelog' => [[1,2],[3,4]],
// ];
2022-11-23 00:01:37 +01:00
2022-11-25 13:30:03 +01:00
// $response = $this->withHeaders([
// 'X-API-SECRET' => config('ninja.api_secret'),
// 'X-API-TOKEN' => $this->token,
// ])->post('/api/v1/tasks', $data);
2022-11-23 00:01:37 +01:00
2022-11-25 13:30:03 +01:00
// $arr = $response->json();
// $response->assertStatus(200);
2022-11-23 00:01:37 +01:00
2022-11-25 13:30:03 +01:00
// $response = $this->withHeaders([
// 'X-API-SECRET' => config('ninja.api_secret'),
// 'X-API-TOKEN' => $this->token,
// ])->putJson('/api/v1/tasks/' . $arr['data']['id'], $data);
2022-11-23 00:01:37 +01:00
2022-11-25 13:30:03 +01:00
// $arr = $response->json();
2022-11-23 00:01:37 +01:00
2022-11-25 13:30:03 +01:00
// $response->assertStatus(200);
2022-11-23 00:01:37 +01:00
2022-11-25 13:30:03 +01:00
// }
2022-11-23 00:01:37 +01:00
2022-08-18 05:29:18 +02:00
public function testTimeLogValidation()
{
$data = [
'timelog' => $this->faker->firstName(),
];
try {
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks', $data);
$arr = $response->json();
} catch (ValidationException $e) {
$response->assertStatus(302);
}
}
public function testTimeLogValidation1()
{
$data = [
'timelog' => [[1,2],[3,4]],
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks', $data);
$arr = $response->json();
$response->assertStatus(200);
}
2022-11-23 00:01:37 +01:00
2022-08-18 05:29:18 +02:00
public function testTimeLogValidation2()
{
$data = [
'timelog' => [],
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks', $data);
$arr = $response->json();
$response->assertStatus(200);
}
public function testTimeLogValidation3()
{
$data = [
'timelog' => [["a","b",'d'],["c","d",'d']],
2022-08-18 05:29:18 +02:00
];
try {
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks', $data);
$arr = $response->json();
} catch (ValidationException $e) {
$response->assertStatus(302);
}
}
public function testTimeLogValidation4()
{
$data = [
'timelog' => [[1,2,'d'],[3,0,'d']],
2022-08-18 05:29:18 +02:00
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks', $data);
$arr = $response->json();
$response->assertStatus(200);
}
2022-03-10 07:17:40 +01:00
public function testStartTask()
{
$log = [
[2, 1,'d'],
[10, 20,'d'],
2022-03-10 07:17:40 +01:00
];
$last = end($log);
$this->assertEquals(10, $last[0]);
$this->assertEquals(20, $last[1]);
$new = [time(), 0];
array_push($log, $new);
$this->assertEquals(3, count($log));
//test task is started
$last = end($log);
$this->assertTrue($last[1] === 0);
//stop task
$last = end($log);
$last[1] = time();
2022-03-10 07:18:17 +01:00
$this->assertTrue($last[1] !== 0);
2022-03-10 07:17:40 +01:00
}
2020-10-12 22:42:02 +02:00
public function testTaskPost()
{
$data = [
'description' => $this->faker->firstName(),
'number' => 'taskynumber',
2022-08-18 05:29:18 +02:00
'client_id' => $this->client->id,
2020-10-12 22:42:02 +02:00
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks', $data);
2020-10-12 22:42:02 +02:00
2020-10-29 11:44:05 +01:00
$arr = $response->json();
2020-10-12 22:42:02 +02:00
$response->assertStatus(200);
2020-10-29 11:44:05 +01:00
2021-03-20 01:25:44 +01:00
$this->assertEquals('taskynumber', $arr['data']['number']);
2022-03-30 04:54:40 +02:00
$this->assertLessThan(5, strlen($arr['data']['time_log']));
2021-03-20 01:25:44 +01:00
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->put('/api/v1/tasks/'.$arr['data']['id'], $data);
2021-03-20 01:25:44 +01:00
$response->assertStatus(200);
try {
$response = $this->withHeaders([
2021-03-20 01:25:44 +01:00
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks', $data);
$arr = $response->json();
} catch (ValidationException $e) {
2021-03-20 01:25:44 +01:00
$response->assertStatus(302);
}
2020-10-29 11:44:05 +01:00
$this->assertNotEmpty($arr['data']['number']);
}
2022-03-30 04:54:40 +02:00
public function testTaskPostNoDefinedTaskNumber()
{
$data = [
'description' => $this->faker->firstName(),
2022-03-30 04:54:40 +02:00
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks', $data);
2022-03-30 04:54:40 +02:00
$arr = $response->json();
$response->assertStatus(200);
$this->assertNotEmpty($arr['data']['number']);
}
2022-08-18 05:29:18 +02:00
public function testTaskWithBadClientId()
{
$data = [
'client_id' => $this->faker->firstName(),
];
try {
2023-02-16 02:36:09 +01:00
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks', $data);
2022-08-18 05:29:18 +02:00
$arr = $response->json();
} catch (ValidationException $e) {
$response->assertStatus(302);
}
}
2020-10-29 11:44:05 +01:00
public function testTaskPostWithActionStart()
{
$data = [
'description' => $this->faker->firstName(),
2020-10-29 11:44:05 +01:00
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks?action=start', $data);
2020-10-29 11:44:05 +01:00
$arr = $response->json();
$response->assertStatus(200);
2020-10-12 22:42:02 +02:00
}
public function testTaskPut()
{
$data = [
'description' => $this->faker->firstName(),
2020-10-12 22:42:02 +02:00
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->put('/api/v1/tasks/'.$this->encodePrimaryKey($this->task->id), $data);
2020-10-12 22:42:02 +02:00
$response->assertStatus(200);
}
2020-10-29 10:40:13 +01:00
public function testTasksGet()
{
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->get('/api/v1/tasks');
2020-10-29 10:40:13 +01:00
$response->assertStatus(200);
}
2020-10-12 22:42:02 +02:00
public function testTaskGet()
{
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->get('/api/v1/tasks/'.$this->encodePrimaryKey($this->task->id));
2020-10-12 22:42:02 +02:00
$response->assertStatus(200);
}
public function testTaskNotArchived()
{
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->get('/api/v1/tasks/'.$this->encodePrimaryKey($this->task->id));
2020-10-12 22:42:02 +02:00
$arr = $response->json();
$this->assertEquals(0, $arr['data']['archived_at']);
}
public function testTaskArchived()
{
$data = [
'ids' => [$this->encodePrimaryKey($this->task->id)],
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks/bulk?action=archive', $data);
2020-10-12 22:42:02 +02:00
$arr = $response->json();
$this->assertNotNull($arr['data'][0]['archived_at']);
}
public function testTaskRestored()
{
$data = [
'ids' => [$this->encodePrimaryKey($this->task->id)],
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks/bulk?action=restore', $data);
2020-10-12 22:42:02 +02:00
$arr = $response->json();
$this->assertEquals(0, $arr['data'][0]['archived_at']);
}
public function testTaskDeleted()
{
$data = [
'ids' => [$this->encodePrimaryKey($this->task->id)],
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks/bulk?action=delete', $data);
2020-10-12 22:42:02 +02:00
$arr = $response->json();
$this->assertTrue($arr['data'][0]['is_deleted']);
}
2022-03-30 04:54:40 +02:00
public function testTaskPostWithStartAction()
{
$data = [
'description' => $this->faker->firstName(),
'number' => 'taskynumber2',
2022-03-30 04:54:40 +02:00
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks?start=true', $data);
2022-03-30 04:54:40 +02:00
$arr = $response->json();
$response->assertStatus(200);
$this->assertEquals('taskynumber2', $arr['data']['number']);
$this->assertGreaterThan(5, strlen($arr['data']['time_log']));
}
public function testTaskPostWithStopAction()
{
$data = [
'description' => $this->faker->firstName(),
2022-03-30 04:54:40 +02:00
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token,
])->post('/api/v1/tasks?stop=true', $data);
2022-03-30 04:54:40 +02:00
$arr = $response->json();
$response->assertStatus(200);
$this->assertLessThan(5, strlen($arr['data']['time_log']));
}
2020-10-12 22:42:02 +02:00
}