mirror of
https://github.com/invoiceninja/invoiceninja.git
synced 2024-11-10 13:12:50 +01:00
655 lines
16 KiB
PHP
655 lines
16 KiB
PHP
<?php
|
|
/**
|
|
* Invoice Ninja (https://invoiceninja.com).
|
|
*
|
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
|
*
|
|
* @copyright Copyright (c) 2021. Invoice Ninja LLC (https://invoiceninja.com)
|
|
*
|
|
* @license https://www.elastic.co/licensing/elastic-license
|
|
*/
|
|
|
|
namespace Tests\Feature;
|
|
|
|
use App\Models\Task;
|
|
use App\Utils\Traits\MakesHash;
|
|
use Illuminate\Database\Eloquent\Model;
|
|
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
|
use Illuminate\Support\Facades\Session;
|
|
use Illuminate\Validation\ValidationException;
|
|
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
|
|
{
|
|
parent::setUp();
|
|
|
|
$this->makeTestData();
|
|
|
|
Session::start();
|
|
|
|
$this->faker = \Faker\Factory::create();
|
|
|
|
Model::reguard();
|
|
}
|
|
|
|
private function checkTimeLog(array $log): bool
|
|
{
|
|
if (count($log) == 0) {
|
|
return true;
|
|
}
|
|
|
|
/*Get first value of all arrays*/
|
|
$result = array_column($log, 0);
|
|
|
|
/*Sort the array in ascending order*/
|
|
asort($result);
|
|
|
|
$new_array = [];
|
|
|
|
/*Rebuild the array in order*/
|
|
foreach ($result as $key => $value) {
|
|
$new_array[] = $log[$key];
|
|
}
|
|
|
|
/*Iterate through the array and perform checks*/
|
|
foreach ($new_array as $key => $array) {
|
|
/*Flag which helps us know if there is a NEXT timelog*/
|
|
$next = false;
|
|
/* If there are more than 1 time log in the array, ensure the last timestamp is not zero*/
|
|
if (count($new_array) >1 && $array[1] == 0) {
|
|
return false;
|
|
}
|
|
|
|
/* 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) {
|
|
return false;
|
|
}
|
|
|
|
/* Find the next time log value - if it exists */
|
|
if (array_key_exists($key+1, $new_array)) {
|
|
$next = $new_array[$key+1];
|
|
}
|
|
|
|
/* check the next time log and ensure the start time is GREATER than the end time of the previous record */
|
|
if ($next && $next[0] < $array[1]) {
|
|
return false;
|
|
}
|
|
|
|
/* Get the last row of the timelog*/
|
|
$last_row = end($new_array);
|
|
|
|
/*If the last value is NOT zero, ensure start time is not GREATER than the endtime */
|
|
if ($last_row[1] != 0 && $last_row[0] > $last_row[1]) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
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));
|
|
}
|
|
|
|
public function testTimeLogChecker8()
|
|
{
|
|
$log = [
|
|
[1,3],
|
|
[50,0]
|
|
];
|
|
|
|
$this->assertTrue($this->checkTimeLog($log));
|
|
}
|
|
|
|
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));
|
|
}
|
|
|
|
|
|
public function testTimeLogChecker12()
|
|
{
|
|
$log = [
|
|
[1,2,'a',true],
|
|
[3,4,'d',false],
|
|
];
|
|
|
|
$this->assertTrue($this->checkTimeLog($log));
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
public function testTaskLockingGate()
|
|
{
|
|
$data = [
|
|
'timelog' => [[1,2,'a'],[3,4,'d']],
|
|
];
|
|
|
|
$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']));
|
|
$task->company->invoice_task_lock = true;
|
|
$task->invoice_id = $this->invoice->id;
|
|
$task->push();
|
|
|
|
$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);
|
|
}
|
|
|
|
|
|
// public function testTaskLocking()
|
|
// {
|
|
// $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);
|
|
|
|
|
|
// $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);
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
|
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);
|
|
}
|
|
|
|
|
|
|
|
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']],
|
|
];
|
|
|
|
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']],
|
|
];
|
|
|
|
$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 testStartTask()
|
|
{
|
|
$log = [
|
|
[2, 1,'d'],
|
|
[10, 20,'d'],
|
|
];
|
|
|
|
$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();
|
|
|
|
$this->assertTrue($last[1] !== 0);
|
|
}
|
|
|
|
public function testTaskPost()
|
|
{
|
|
$data = [
|
|
'description' => $this->faker->firstName(),
|
|
'number' => 'taskynumber',
|
|
'client_id' => $this->client->id,
|
|
];
|
|
|
|
$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);
|
|
|
|
$this->assertEquals('taskynumber', $arr['data']['number']);
|
|
$this->assertLessThan(5, strlen($arr['data']['time_log']));
|
|
|
|
$response = $this->withHeaders([
|
|
'X-API-SECRET' => config('ninja.api_secret'),
|
|
'X-API-TOKEN' => $this->token,
|
|
])->put('/api/v1/tasks/'.$arr['data']['id'], $data);
|
|
|
|
$response->assertStatus(200);
|
|
|
|
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);
|
|
}
|
|
|
|
$this->assertNotEmpty($arr['data']['number']);
|
|
}
|
|
|
|
public function testTaskPostNoDefinedTaskNumber()
|
|
{
|
|
$data = [
|
|
'description' => $this->faker->firstName(),
|
|
];
|
|
|
|
$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);
|
|
$this->assertNotEmpty($arr['data']['number']);
|
|
}
|
|
|
|
public function testTaskWithBadClientId()
|
|
{
|
|
$data = [
|
|
'client_id' => $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 testTaskPostWithActionStart()
|
|
{
|
|
$data = [
|
|
'description' => $this->faker->firstName(),
|
|
];
|
|
|
|
$response = $this->withHeaders([
|
|
'X-API-SECRET' => config('ninja.api_secret'),
|
|
'X-API-TOKEN' => $this->token,
|
|
])->post('/api/v1/tasks?action=start', $data);
|
|
|
|
$arr = $response->json();
|
|
$response->assertStatus(200);
|
|
}
|
|
|
|
public function testTaskPut()
|
|
{
|
|
$data = [
|
|
'description' => $this->faker->firstName(),
|
|
];
|
|
|
|
$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);
|
|
|
|
$response->assertStatus(200);
|
|
}
|
|
|
|
public function testTasksGet()
|
|
{
|
|
$response = $this->withHeaders([
|
|
'X-API-SECRET' => config('ninja.api_secret'),
|
|
'X-API-TOKEN' => $this->token,
|
|
])->get('/api/v1/tasks');
|
|
|
|
$response->assertStatus(200);
|
|
}
|
|
|
|
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));
|
|
|
|
$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));
|
|
|
|
$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);
|
|
|
|
$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);
|
|
|
|
$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);
|
|
|
|
$arr = $response->json();
|
|
|
|
$this->assertTrue($arr['data'][0]['is_deleted']);
|
|
}
|
|
|
|
public function testTaskPostWithStartAction()
|
|
{
|
|
$data = [
|
|
'description' => $this->faker->firstName(),
|
|
'number' => 'taskynumber2',
|
|
];
|
|
|
|
$response = $this->withHeaders([
|
|
'X-API-SECRET' => config('ninja.api_secret'),
|
|
'X-API-TOKEN' => $this->token,
|
|
])->post('/api/v1/tasks?start=true', $data);
|
|
|
|
$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(),
|
|
];
|
|
|
|
$response = $this->withHeaders([
|
|
'X-API-SECRET' => config('ninja.api_secret'),
|
|
'X-API-TOKEN' => $this->token,
|
|
])->post('/api/v1/tasks?stop=true', $data);
|
|
|
|
$arr = $response->json();
|
|
$response->assertStatus(200);
|
|
|
|
$this->assertLessThan(5, strlen($arr['data']['time_log']));
|
|
}
|
|
}
|