.
 */
declare(strict_types=1);
namespace Tests\Feature\Controllers\Admin;
use Carbon\Carbon;
use FireflyConfig;
use FireflyIII\Exceptions\FireflyException;
use FireflyIII\Models\Configuration;
use FireflyIII\Repositories\User\UserRepositoryInterface;
use FireflyIII\Services\Github\Object\Release;
use FireflyIII\Services\Github\Request\UpdateRequest;
use Log;
use Mockery;
use Tests\TestCase;
/**
 * Class UpdateControllerTest
 */
class UpdateControllerTest extends TestCase
{
    /**
     *
     */
    public function setUp(): void
    {
        parent::setUp();
        Log::info(sprintf('Now in %s.', get_class($this)));
    }
    /**
     * @covers \FireflyIII\Http\Controllers\Admin\UpdateController
     */
    public function testIndex(): void
    {
        // mock stuff
        $userRepos = $this->mock(UserRepositoryInterface::class);
        // mock calls
        $userRepos->shouldReceive('hasRole')->withArgs([Mockery::any(), 'owner'])->andReturn(true)->atLeast()->once();
        $this->mockDefaultSession();
        // mock update calls.
        $config            = new Configuration;
        $config->data      = -1;
        FireflyConfig::shouldReceive('get')->withArgs(['permission_update_check', -1])->once()->andReturn($config);
        // call service
        $this->be($this->user());
        $response = $this->get(route('admin.update-check'));
        $response->assertStatus(200);
        $response->assertSee('
');
    }
    /**
     * @covers \FireflyIII\Http\Controllers\Admin\UpdateController
     */
    public function testPost(): void
    {
        // mock stuff
        $userRepos = $this->mock(UserRepositoryInterface::class);
        // mock calls
        $userRepos->shouldReceive('hasRole')->withArgs([Mockery::any(), 'owner'])->andReturn(true)->atLeast()->once();
        $userRepos->shouldReceive('hasRole')->withArgs([Mockery::any(), 'demo'])->andReturn(false)->atLeast()->once();
        $this->mockDefaultSession();
        // mock update calls
        FireflyConfig::shouldReceive('set')->withArgs(['permission_update_check', 1])->once()->andReturn(new Configuration);
        FireflyConfig::shouldReceive('set')->withArgs(['last_update_check', Mockery::any()])->once()->andReturn(new Configuration);
        // call service
        $this->be($this->user());
        $response = $this->post(route('admin.update-check.post'), ['check_for_updates' => 1]);
        $response->assertSessionHas('success');
        $response->assertStatus(302);
        $response->assertRedirect(route('admin.update-check'));
    }
    /**
     * @covers \FireflyIII\Http\Controllers\Admin\UpdateController
     * @covers \FireflyIII\Helpers\Update\UpdateTrait
     */
    public function testUpdateCheck(): void
    {
        // mock stuff
        $userRepos = $this->mock(UserRepositoryInterface::class);
        // mock calls
        $userRepos->shouldReceive('hasRole')->withArgs([Mockery::any(), 'owner'])->andReturn(true)->atLeast()->once();
        $userRepos->shouldReceive('hasRole')->withArgs([Mockery::any(), 'demo'])->andReturn(false)->atLeast()->once();
        FireflyConfig::shouldReceive('set')->withArgs(['last_update_check', Mockery::any()])->once()->andReturn(new Configuration);
        $this->mockDefaultSession();
        // set some data
        $version = config('firefly.version');
        $date    = new Carbon;
        $date->subDays(5);
        $releases = [
            new Release(['id' => 'x', 'title' => $version . '.1', 'content' => '', 'updated' => $date]),
        ];
        $updater  = $this->mock(UpdateRequest::class);
        $updater->shouldReceive('call')->andReturnNull();
        $updater->shouldReceive('getReleases')->andReturn($releases);
        $this->be($this->user());
        $response = $this->post(route('admin.update-check.manual'));
        $response->assertStatus(200);
        $response->assertSee($version);
        $response->assertSee('which was released on');
        $response->assertSee($version . '.1');
    }
    /**
     * @covers \FireflyIII\Http\Controllers\Admin\UpdateController
     * @covers \FireflyIII\Helpers\Update\UpdateTrait
     */
    public function testUpdateCheckCurrent(): void
    {
        $userRepos = $this->mock(UserRepositoryInterface::class);
        $userRepos->shouldReceive('hasRole')->withArgs([Mockery::any(), 'owner'])->andReturn(true)->atLeast()->once();
        $userRepos->shouldReceive('hasRole')->withArgs([Mockery::any(), 'demo'])->andReturn(false)->atLeast()->once();
        $this->mockDefaultSession();
        FireflyConfig::shouldReceive('set')->withArgs(['last_update_check', Mockery::any()])->once()->andReturn(new Configuration);
        $date = new Carbon;
        $date->subDays(5);
        $version  = config('firefly.version');
        $releases = [
            new Release(['id' => 'x', 'title' => $version, 'content' => '', 'updated' => $date]),
        ];
        $updater  = $this->mock(UpdateRequest::class);
        $updater->shouldReceive('call')->andReturnNull();
        $updater->shouldReceive('getReleases')->andReturn($releases);
        $this->be($this->user());
        $response = $this->post(route('admin.update-check.manual'));
        $response->assertStatus(200);
        $response->assertSee($version);
        $response->assertSee('the latest available release');
    }
    /**
     * @covers \FireflyIII\Http\Controllers\Admin\UpdateController
     * @covers \FireflyIII\Helpers\Update\UpdateTrait
     */
    public function testUpdateCheckError(): void
    {
        $userRepos = $this->mock(UserRepositoryInterface::class);
        $userRepos->shouldReceive('hasRole')->withArgs([Mockery::any(), 'owner'])->andReturn(true)->atLeast()->once();
        $userRepos->shouldReceive('hasRole')->withArgs([Mockery::any(), 'demo'])->andReturn(false)->atLeast()->once();
        $this->mockDefaultSession();
        FireflyConfig::shouldReceive('set')->withArgs(['last_update_check', Mockery::any()])->once()->andReturn(new Configuration);
        $releases = [];
        $updater  = $this->mock(UpdateRequest::class);
        $updater->shouldReceive('call')->andThrow(FireflyException::class, 'Something broke.');
        $updater->shouldReceive('getReleases')->andReturn($releases);
        $this->be($this->user());
        $response = $this->post(route('admin.update-check.manual'));
        $response->assertStatus(200);
        $response->assertSee('An error occurred while checking');
    }
    /**
     * @covers \FireflyIII\Http\Controllers\Admin\UpdateController
     * @covers \FireflyIII\Helpers\Update\UpdateTrait
     */
    public function testUpdateCheckNewer(): void
    {
        $userRepos = $this->mock(UserRepositoryInterface::class);
        $userRepos->shouldReceive('hasRole')->withArgs([Mockery::any(), 'owner'])->andReturn(true)->atLeast()->once();
        $userRepos->shouldReceive('hasRole')->withArgs([Mockery::any(), 'demo'])->andReturn(false)->atLeast()->once();
        $this->mockDefaultSession();
        FireflyConfig::shouldReceive('set')->withArgs(['last_update_check', Mockery::any()])->once()->andReturn(new Configuration);
        $version  = config('firefly.version') . '-alpha';
        $releases = [
            new Release(['id' => 'x', 'title' => $version, 'content' => '', 'updated' => new Carbon]),
        ];
        $updater  = $this->mock(UpdateRequest::class);
        $updater->shouldReceive('call')->andReturnNull();
        $updater->shouldReceive('getReleases')->andReturn($releases);
        // expect a new release (because of .1)
        $this->be($this->user());
        $response = $this->post(route('admin.update-check.manual'));
        $response->assertStatus(200);
        $response->assertSee($version);
        $response->assertSee('which is newer than the');
    }
}