blessing-skin-server/tests/ServicesTest/PluginManagerTest.php
2020-08-22 15:10:33 +08:00

740 lines
27 KiB
PHP

<?php
namespace Tests;
use App\Events;
use App\Services\Option;
use App\Services\Plugin;
use App\Services\PluginManager;
use Event;
use Illuminate\Filesystem\Filesystem;
use ReflectionClass;
class PluginManagerTest extends TestCase
{
public function testPreventBootingAgain()
{
$this->mock(Filesystem::class, function ($mock) {
$mock->shouldReceive('directories')->times(0);
});
resolve(PluginManager::class)->boot();
}
public function testNotLoadDisabled()
{
$dir = config('plugins.directory');
config(['plugins.directory' => base_path('tests/__mocks__')]);
app()->forgetInstance(PluginManager::class);
resolve(PluginManager::class)->boot();
$this->assertFalse(class_exists('Fake\Faker'));
config(['plugins.directory' => $dir]);
}
public function testNotLoadUnsatisfied()
{
$this->mock(Option::class, function ($mock) {
$mock->shouldReceive('get')->with('plugins_enabled', '[]')->andReturn('[]');
});
$this->mock(Filesystem::class, function ($mock) {
$mock->shouldReceive('directories')
->with(base_path('plugins'))
->once()
->andReturn(collect(['/nano']));
$mock->shouldReceive('exists')
->with('/nano'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(true);
$mock->shouldReceive('get')
->with('/nano'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(json_encode([
'name' => 'fake',
'version' => '0.0.0',
'require' => ['blessing-skin-server' => '0.0.0'],
]));
$mock->shouldNotReceive('getRequire');
});
app()->forgetInstance(PluginManager::class);
resolve(PluginManager::class)->boot();
}
public function testReportDuplicatedPlugins()
{
$this->mock(Option::class, function ($mock) {
$mock->shouldReceive('get')->with('plugins_enabled', '[]')->andReturn('[]');
});
$this->mock(Filesystem::class, function ($mock) {
$mock->shouldReceive('directories')
->with(base_path('plugins'))
->once()
->andReturn(collect(['/nano', '/yuko']));
$mock->shouldReceive('exists')
->with('/nano'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(true);
$mock->shouldReceive('get')
->with('/nano'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(json_encode([
'name' => 'fake',
'version' => '0.0.0',
]));
$mock->shouldReceive('exists')
->with('/yuko'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(true);
$mock->shouldReceive('get')
->with('/yuko'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(json_encode([
'name' => 'fake',
'version' => '0.0.0',
]));
});
$this->expectExceptionMessage(trans('errors.plugins.duplicate', [
'dir1' => '/nano',
'dir2' => '/yuko',
]));
app()->forgetInstance(PluginManager::class);
resolve(PluginManager::class)->boot();
}
public function testDetectVersionChanged()
{
Event::fake();
$this->mock(Option::class, function ($mock) {
$mock->shouldReceive('get')
->with('plugins_enabled', '[]')
->andReturn(json_encode([['name' => 'mayaka', 'version' => '0.0.0']]));
$mock->shouldReceive('set')
->with(
'plugins_enabled',
json_encode([['name' => 'mayaka', 'version' => '0.1.0']])
)
->once();
});
$this->mock(Filesystem::class, function ($mock) {
$mock->shouldReceive('directories')
->with(base_path('plugins'))
->once()
->andReturn(collect(['/mayaka']));
$mock->shouldReceive('exists')
->with('/mayaka'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(true);
$mock->shouldReceive('get')
->with('/mayaka'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(json_encode([
'name' => 'mayaka',
'version' => '0.1.0',
]));
$mock->shouldReceive('exists')
->with('/mayaka/vendor/autoload.php')
->once()
->andReturn(false);
$mock->shouldReceive('exists')
->with('/mayaka/bootstrap.php')
->once()
->andReturn(false);
});
app()->forgetInstance(PluginManager::class);
resolve(PluginManager::class)->boot();
Event::assertDispatched('plugin.versionChanged', function ($eventName, $payload) {
$this->assertEquals('0.1.0', $payload[0]->version);
return true;
});
}
public function testLoadComposer()
{
$this->mock(Option::class, function ($mock) {
$mock->shouldReceive('get')
->with('plugins_enabled', '[]')
->andReturn(json_encode([['name' => 'mayaka', 'version' => '0.0.0']]));
});
$this->mock(Filesystem::class, function ($mock) {
$mock->shouldReceive('directories')
->with(base_path('plugins'))
->once()
->andReturn(collect(['/mayaka']));
$mock->shouldReceive('exists')
->with('/mayaka'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(true);
$mock->shouldReceive('get')
->with('/mayaka'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(json_encode([
'name' => 'mayaka',
'version' => '0.0.0',
]));
$mock->shouldReceive('exists')
->with('/mayaka/vendor/autoload.php')
->once()
->andReturn(true);
$mock->shouldReceive('getRequire')
->with('/mayaka/vendor/autoload.php')
->once();
$mock->shouldReceive('exists')
->with('/mayaka/bootstrap.php')
->once()
->andReturn(false);
});
app()->forgetInstance(PluginManager::class);
resolve(PluginManager::class)->boot();
}
public function testLoadViews()
{
$dir = config('plugins.directory');
config(['plugins.directory' => base_path('tests/__mocks__')]);
$this->mock(Option::class, function ($mock) {
$mock->shouldReceive('get')
->with('plugins_enabled', '[]')
->andReturn(json_encode([
['name' => 'fake-with-views', 'version' => '0.0.0'],
]));
});
app()->forgetInstance(PluginManager::class);
resolve(PluginManager::class)->boot();
$this->assertTrue(view()->exists('FakeWithViews::example'));
config(['plugins.directory' => $dir]);
}
public function testLoadTranslations()
{
$this->mock(Option::class, function ($mock) {
$mock->shouldReceive('get')
->with('plugins_enabled', '[]')
->andReturn(json_encode([['name' => 'mayaka', 'version' => '0.0.0']]));
});
$this->mock(Filesystem::class, function ($mock) {
$mock->shouldReceive('directories')
->with(base_path('plugins'))
->once()
->andReturn(collect(['/mayaka', '/chitanda']));
$mock->shouldReceive('exists')
->with('/mayaka'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(true);
$mock->shouldReceive('get')
->with('/mayaka'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(json_encode([
'name' => 'mayaka',
'version' => '0.0.0',
'namespace' => 'Mayaka',
]));
$mock->shouldReceive('exists')
->with('/mayaka/vendor/autoload.php')
->once()
->andReturn(false);
$mock->shouldReceive('exists')
->with('/mayaka/bootstrap.php')
->once()
->andReturn(false);
$mock->shouldReceive('exists')
->with('/chitanda'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(true);
$mock->shouldReceive('get')
->with('/chitanda'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(json_encode([
'name' => 'chitanda',
'version' => '0.0.0',
'namespace' => 'Chitanda',
]));
});
$this->instance('translation.loader', \Mockery::mock(\App\Services\Translations\Loader::class, function ($mock) {
$mock->shouldReceive('addNamespace')
->withArgs(['Mayaka', '/mayaka/lang'])
->once();
$mock->shouldReceive('addNamespace')
->withArgs(['Chitanda', '/chitanda/lang'])
->once();
}));
app()->forgetInstance(PluginManager::class);
resolve(PluginManager::class)->boot();
}
public function testLoadBootstrapper()
{
$this->mock(Option::class, function ($mock) {
$mock->shouldReceive('get')
->with('plugins_enabled', '[]')
->andReturn(json_encode([['name' => 'mayaka', 'version' => '0.0.0']]));
});
$this->mock(Filesystem::class, function ($mock) {
$mock->shouldReceive('directories')
->with(base_path('plugins'))
->once()
->andReturn(collect(['/mayaka']));
$mock->shouldReceive('exists')
->with('/mayaka'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(true);
$mock->shouldReceive('get')
->with('/mayaka'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(json_encode([
'name' => 'mayaka',
'version' => '0.0.0',
]));
$mock->shouldReceive('exists')
->with('/mayaka/vendor/autoload.php')
->once()
->andReturn(false);
$mock->shouldReceive('exists')
->with('/mayaka/bootstrap.php')
->once()
->andReturn(true);
$mock->shouldReceive('getRequire')
->with('/mayaka/bootstrap.php')
->once()
->andReturn(function (\Illuminate\Contracts\Events\Dispatcher $events, Plugin $plugin) {
$this->assertTrue(method_exists($events, 'listen'));
$this->assertEquals('mayaka', $plugin->name);
});
});
app()->forgetInstance(PluginManager::class);
resolve(PluginManager::class)->boot();
}
public function testHandleBootstrapperExceptions()
{
Event::fake();
$this->mock(Option::class, function ($mock) {
$mock->shouldReceive('get')
->with('plugins_enabled', '[]')
->andReturn(json_encode([['name' => 'mayaka', 'version' => '0.0.0']]));
});
$this->mock(Filesystem::class, function ($mock) {
$mock->shouldReceive('directories')
->with(base_path('plugins'))
->andReturn(collect(['/mayaka']));
$mock->shouldReceive('exists')
->with('/mayaka'.DIRECTORY_SEPARATOR.'package.json')
->andReturn(true);
$mock->shouldReceive('get')
->with('/mayaka'.DIRECTORY_SEPARATOR.'package.json')
->andReturn(json_encode([
'name' => 'mayaka',
'version' => '0.0.0',
]));
$mock->shouldReceive('exists')
->with('/mayaka/vendor/autoload.php')
->andReturn(false);
$mock->shouldReceive('exists')
->with('/mayaka/bootstrap.php')
->andReturn(true);
$mock->shouldReceive('getRequire')
->with('/mayaka/bootstrap.php')
->andReturn(function () {
throw new \Exception();
}, function () {
abort(500);
});
});
app()->forgetInstance(PluginManager::class);
resolve(PluginManager::class)->boot();
Event::assertDispatched(Events\PluginBootFailed::class, function ($event) {
$this->assertEquals('mayaka', $event->plugin->name);
return true;
});
app()->forgetInstance(PluginManager::class);
$this->expectException(\Symfony\Component\HttpKernel\Exception\HttpException::class);
resolve(PluginManager::class)->boot();
}
public function testLifecycleHooks()
{
$this->mock(Option::class, function ($mock) {
$mock->shouldReceive('get')->with('plugins_enabled', '[]')->andReturn('[]');
});
$this->mock(Filesystem::class, function ($mock) {
$mock->shouldReceive('directories')
->with(base_path('plugins'))
->once()
->andReturn(collect(['/mayaka']));
$mock->shouldReceive('exists')
->with('/mayaka'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(true);
$mock->shouldReceive('get')
->with('/mayaka'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(json_encode([
'name' => 'mayaka',
'version' => '0.0.0',
'namespace' => 'Mayaka',
]));
$mock->shouldReceive('exists')
->with('/mayaka/callbacks.php')
->once()
->andReturn(true);
$mock->shouldReceive('getRequire')
->with('/mayaka/callbacks.php')
->once()
->andReturn([
\App\Events\PluginWasDeleted::class => function ($plugin) {
$this->assertInstanceOf(Plugin::class, $plugin);
$this->assertEquals('mayaka', $plugin->name);
},
]);
});
app()->forgetInstance(PluginManager::class);
resolve(PluginManager::class)->boot();
event(new \App\Events\PluginWasDeleted(new Plugin('/mayaka', ['name' => 'mayaka'])));
}
public function testRegisterAutoload()
{
$dir = config('plugins.directory');
config(['plugins.directory' => base_path('tests/__mocks__')]);
$this->mock(Option::class, function ($mock) {
$mock->shouldReceive('get')
->with('plugins_enabled', '[]')
->andReturn(json_encode([['name' => 'fake', 'version' => '0.0.0']]));
});
$this->assertFalse(class_exists('Fake\Faker'));
app()->forgetInstance(PluginManager::class);
resolve(PluginManager::class)->boot();
$this->assertTrue(class_exists('Fake\FakeServiceProvider'));
config(['plugins.directory' => $dir]);
}
public function testRegisterServiceProviders()
{
Event::fake();
$dir = config('plugins.directory');
config(['plugins.directory' => base_path('tests/__mocks__')]);
$this->mock(Option::class, function ($mock) {
$mock->shouldReceive('get')
->with('plugins_enabled', '[]')
->andReturn(json_encode([['name' => 'fake', 'version' => '0.0.0']]));
});
app()->forgetInstance(PluginManager::class);
resolve(PluginManager::class)->boot();
Event::assertDispatched('provider.loaded');
config(['plugins.directory' => $dir]);
}
public function testGetUnsatisfied()
{
$manager = resolve(PluginManager::class);
$plugin = new Plugin('', ['require' => ['blessing-skin-server' => '^0.0.0']]);
$info = $manager->getUnsatisfied($plugin)->get('blessing-skin-server');
$this->assertEquals(config('app.version'), $info['version']);
$this->assertEquals('^0.0.0', $info['constraint']);
preg_match('/(\d+\.\d+\.\d+)/', PHP_VERSION, $matches);
$version = $matches[1];
$plugin = new Plugin('', ['require' => ['php' => '^0.0.0']]);
$info = $manager->getUnsatisfied($plugin)->get('php');
$this->assertEquals($version, $info['version']);
$this->assertEquals('^0.0.0', $info['constraint']);
$plugin = new Plugin('', ['require' => ['another-plugin' => '0.0.*']]);
$info = $manager->getUnsatisfied($plugin)->get('another-plugin');
$this->assertNull($info['version']);
$this->assertEquals('0.0.*', $info['constraint']);
$reflection = new ReflectionClass($manager);
$property = $reflection->getProperty('enabled');
$property->setAccessible(true);
$property->setValue($manager, collect(['another-plugin' => ['version' => '1.2.3']]));
$info = $manager->getUnsatisfied($plugin)->get('another-plugin');
$this->assertEquals('1.2.3', $info['version']);
$this->assertEquals('0.0.*', $info['constraint']);
$plugin = new Plugin('', ['require' => ['another-plugin' => '^1.0.0']]);
$this->assertFalse($manager->getUnsatisfied($plugin)->has('another-plugin'));
}
public function testGetConflicts()
{
$manager = resolve(PluginManager::class);
$plugin = new Plugin('/', ['enchants' => ['conflicts' => ['a' => '*', 'b' => '^1.2.0']]]);
$reflection = new ReflectionClass($manager);
$property = $reflection->getProperty('enabled');
$property->setAccessible(true);
$property->setValue($manager, collect(['b' => ['version' => '1.2.3']]));
$conflicts = $manager->getConflicts($plugin);
$this->assertNull($conflicts->get('a'));
$info = $conflicts->get('b');
$this->assertEquals('1.2.3', $info['version']);
$this->assertEquals('^1.2.0', $info['constraint']);
$plugin = new Plugin('/', ['enchants' => ['conflicts' => ['b' => '^0.0.0']]]);
$this->assertNull($manager->getConflicts($plugin)->get('b'));
}
public function testFormatUnresolved()
{
app()->forgetInstance(PluginManager::class);
$manager = resolve(PluginManager::class);
$reflection = new ReflectionClass($manager);
$property = $reflection->getProperty('plugins');
$property->setAccessible(true);
$property->setValue($manager, collect([
'dep' => new Plugin('', ['title' => 'dep', 'version' => '0.0.0']),
'conf' => new Plugin('', ['title' => 'conf', 'version' => '1.2.3']),
]));
$unsatisfied = collect([
'blessing-skin-server' => ['version' => '4.0.0', 'constraint' => '^5.0.0'],
'php' => ['version' => '7.2.0', 'constraint' => '^7.3.0'],
'dep' => ['version' => '0.0.0', 'constraint' => '^6.6.6'],
'whatever' => ['version' => null, 'constraint' => '^1.2.3'],
]);
$conflicts = collect([
'conf' => ['version' => '1.2.3', 'constraint' => '^1.0.0'],
]);
$received = $manager->formatUnresolved($unsatisfied, $conflicts);
$expected = [
trans('admin.plugins.operations.unsatisfied.version', [
'title' => 'Blessing Skin Server',
'constraint' => '^5.0.0',
]),
trans('admin.plugins.operations.unsatisfied.version', [
'title' => 'PHP',
'constraint' => '^7.3.0',
]),
trans('admin.plugins.operations.unsatisfied.version', [
'title' => 'dep',
'constraint' => '^6.6.6',
]),
trans('admin.plugins.operations.unsatisfied.disabled', ['name' => 'whatever']),
trans('admin.plugins.operations.unsatisfied.conflict', ['title' => 'conf']),
];
$this->assertEquals($expected, $received);
}
public function testEnable()
{
Event::fake();
app()->forgetInstance(PluginManager::class);
$manager = resolve(PluginManager::class);
$reflection = new ReflectionClass($manager);
$property = $reflection->getProperty('plugins');
$property->setAccessible(true);
$property->setValue($manager, collect([
'fake' => new Plugin('', ['name' => 'fake']),
'dep' => new Plugin('', ['name' => 'dep', 'require' => ['a' => '*']]),
]));
$this->assertFalse($manager->enable('nope'));
$this->assertTrue($manager->enable('fake'));
// re-enable should be allowed
$this->assertTrue($manager->enable('fake'));
Event::assertDispatched(Events\PluginWasEnabled::class, function ($event) {
$this->assertEquals('fake', $event->plugin->name);
return true;
});
$this->assertTrue($manager->getEnabledPlugins()->has('fake'));
$this->assertEquals(
'fake',
json_decode(resolve(\App\Services\Option::class)->get('plugins_enabled'), true)[0]['name']
);
$this->assertTrue($manager->enable('dep')['unsatisfied']->isNotEmpty());
}
public function testDisable()
{
Event::fake();
app()->forgetInstance(PluginManager::class);
$manager = resolve(PluginManager::class);
$reflection = new ReflectionClass($manager);
$property = $reflection->getProperty('plugins');
$property->setAccessible(true);
$plugin = new Plugin('', ['name' => 'fake']);
$plugin->setEnabled(true);
$property->setValue($manager, collect(['fake' => $plugin]));
$manager->disable('fake');
Event::assertDispatched(Events\PluginWasDisabled::class, function ($event) {
$this->assertEquals('fake', $event->plugin->name);
return true;
});
$this->assertFalse($manager->getEnabledPlugins()->has('fake'));
$this->assertCount(0, json_decode(resolve(\App\Services\Option::class)->get('plugins_enabled'), true));
}
public function testDelete()
{
Event::fake();
$this->mock(Option::class, function ($mock) {
$mock->shouldReceive('get')
->with('plugins_enabled', '[]')
->andReturn(json_encode([['name' => 'fake', 'version' => '0.0.0']]));
$mock->shouldReceive('set')
->with('plugins_enabled', '[]');
});
$this->mock(Filesystem::class, function ($mock) {
$mock->shouldReceive('directories')->andReturn(collect([]));
$mock->shouldReceive('deleteDirectory')->with('/fake')->once();
});
app()->forgetInstance(PluginManager::class);
$manager = resolve(PluginManager::class);
$reflection = new ReflectionClass($manager);
$property = $reflection->getProperty('plugins');
$property->setAccessible(true);
$plugin = new Plugin('/fake', ['name' => 'fake']);
$plugin->setEnabled(true);
$property->setValue($manager, collect(['fake' => $plugin]));
$manager->delete('fake');
Event::assertDispatched(Events\PluginWasDisabled::class, function ($event) {
$this->assertEquals('fake', $event->plugin->name);
return true;
});
Event::assertDispatched(Events\PluginWasDeleted::class, function ($event) {
$this->assertEquals('fake', $event->plugin->name);
return true;
});
$this->assertFalse($manager->getEnabledPlugins()->has('fake'));
$this->assertTrue($manager->all()->isEmpty());
}
public function testHelpers()
{
$manager = app('plugins');
$reflection = new ReflectionClass($manager);
$property = $reflection->getProperty('plugins');
$property->setAccessible(true);
$property->setValue($manager, collect(['fake' => new Plugin('', ['name' => 'fake', 'version' => '1'])]));
$this->assertNull(plugin('nope'));
$this->assertInstanceOf(Plugin::class, plugin('fake'));
$this->assertEquals(
url('plugins').'/fake/assets/relative?v=1',
plugin_assets('fake', 'relative')
);
$this->expectExceptionMessage('No such plugin.');
plugin_assets('nope', 'relative');
}
public function testDefaultPluginsDirectory()
{
$old = config('plugins.directory');
config(['plugins.directory' => null]);
$directories = app('plugins')->getPluginsDirs();
$this->assertEquals(1, $directories->count());
$this->assertEquals(base_path('plugins'), $directories->first());
config(['plugins.directory' => $old]);
}
public function testReadMultipleDirectories()
{
$old = config('plugins.directory');
config(['plugins.directory' => '/kumiko,/reina']);
$this->mock(Option::class, function ($mock) {
$mock->shouldReceive('get')->with('plugins_enabled', '[]')->andReturn('[]');
});
$this->mock(Filesystem::class, function ($mock) {
$mock->shouldReceive('directories')
->with('/kumiko')
->once()
->andReturn(collect(['/a', '/b']));
$mock->shouldReceive('directories')
->with('/reina')
->once()
->andReturn(collect(['/b', '/c']));
$mock->shouldReceive('exists')
->with('/a'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(false);
$mock->shouldReceive('exists')
->with('/b'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(false);
$mock->shouldReceive('exists')
->with('/c'.DIRECTORY_SEPARATOR.'package.json')
->once()
->andReturn(false);
});
app()->forgetInstance(PluginManager::class);
resolve(PluginManager::class)->all();
config(['plugins.directory' => $old]);
}
}