Files
Grav-Admin-Plugin/admin.php

717 lines
23 KiB
PHP
Raw Normal View History

2014-08-05 13:06:38 -07:00
<?php
namespace Grav\Plugin;
2015-12-10 12:07:36 +01:00
use Grav\Common\File\CompiledYamlFile;
use Grav\Common\GPM\GPM;
use Grav\Common\Grav;
use Grav\Common\Inflector;
use Grav\Common\Language\Language;
use Grav\Common\Page\Page;
use Grav\Common\Page\Pages;
use Grav\Common\Plugin;
use Grav\Common\Uri;
2015-12-10 12:07:36 +01:00
use Grav\Common\User\User;
2014-10-01 22:28:16 +03:00
use RocketTheme\Toolbox\File\File;
use RocketTheme\Toolbox\Event\Event;
2014-10-01 22:28:16 +03:00
use RocketTheme\Toolbox\Session\Session;
2014-08-05 13:06:38 -07:00
class AdminPlugin extends Plugin
{
2016-01-22 10:42:38 +02:00
public $features = [
'blueprints' => 1000,
];
2014-08-05 13:06:38 -07:00
/**
* @var bool
*/
protected $active = false;
/**
* @var string
*/
protected $template;
2014-09-03 22:22:03 -06:00
/**
* @var string
*/
protected $theme;
2014-08-05 13:06:38 -07:00
/**
* @var string
*/
protected $route;
/**
* @var Uri
*/
protected $uri;
/**
* @var Admin
*/
protected $admin;
2014-10-01 22:28:16 +03:00
/**
* @var Session
*/
protected $session;
/**
* @var Popularity
*/
protected $popularity;
2014-10-01 22:28:16 +03:00
/**
* @var string
*/
protected $base;
/**
* @return array
*/
public static function getSubscribedEvents()
{
if (!Grav::instance()['config']->get('plugins.admin-pro.enabled')) {
return [
'onPluginsInitialized' => [['setup', 100000], ['onPluginsInitialized', 1001]],
2015-12-10 12:07:36 +01:00
'onShutdown' => ['onShutdown', 1000],
'onFormProcessed' => ['onFormProcessed', 0],
'onAdminDashboard' => ['onAdminDashboard', 0],
];
2015-12-10 12:07:36 +01:00
}
2015-12-10 22:28:30 +01:00
return [];
2015-12-10 12:07:36 +01:00
}
2015-12-10 11:02:19 -07:00
/**
* If the admin path matches, initialize the Login plugin configuration and set the admin
* as active.
*/
public function setup()
{
$route = $this->config->get('plugins.admin.route');
if (!$route) {
return;
}
$this->base = '/' . trim($route, '/');
$this->uri = $this->grav['uri'];
2015-12-10 11:02:19 -07:00
// check for existence of a user account
$account_dir = $file_path = $this->grav['locator']->findResource('account://');
$user_check = glob($account_dir . '/*.yaml');
2015-12-10 11:02:19 -07:00
// If no users found, go to register
if ($user_check == false || count((array)$user_check) == 0) {
if (!$this->isAdminPath()) {
$this->grav->redirect($this->base);
2015-12-10 11:02:19 -07:00
}
$this->template = 'register';
2015-12-10 11:02:19 -07:00
}
// Only activate admin if we're inside the admin path.
if ($this->isAdminPath()) {
2015-12-10 11:02:19 -07:00
$this->active = true;
}
}
2015-12-10 12:07:36 +01:00
/**
* Validate a value. Currently validates
*
* - 'user' for username format and username availability.
* - 'password1' for password format
* - 'password2' for equality to password1
*
* @param string $type The field type
* @param string $value The field value
* @param string $extra Any extra value required
*
2016-01-21 09:46:38 +02:00
* @return bool
2015-12-10 12:07:36 +01:00
*/
protected function validate($type, $value, $extra = '')
{
switch ($type) {
case 'username_format':
if (!preg_match('/^[a-z0-9_-]{3,16}$/', $value)) {
return false;
}
return true;
case 'password1':
if (!preg_match('/(?=.*\d)(?=.*[a-z])(?=.*[A-Z]).{8,}/', $value)) {
return false;
}
return true;
case 'password2':
if (strcmp($value, $extra)) {
return false;
}
return true;
}
2016-01-21 09:46:38 +02:00
return false;
2015-12-10 12:07:36 +01:00
}
/**
* Process the admin registration form.
*
* @param Event $event
*/
public function onFormProcessed(Event $event)
{
$form = $event['form'];
$action = $event['action'];
switch ($action) {
case 'register_admin_user':
if (!$this->config->get('plugins.login.enabled')) {
throw new \RuntimeException($this->grav['language']->translate('PLUGIN_LOGIN.PLUGIN_LOGIN_DISABLED'));
}
$data = [];
$username = $form->value('username');
if ($form->value('password1') != $form->value('password2')) {
2015-12-10 12:07:36 +01:00
$this->grav->fireEvent('onFormValidationError',
new Event([
'form' => $form,
'message' => $this->grav['language']->translate('PLUGIN_LOGIN.PASSWORDS_DO_NOT_MATCH')
]));
$event->stopPropagation();
return;
}
2015-12-10 12:07:36 +01:00
$data['password'] = $form->value('password1');
$fields = [
'email',
'fullname',
'title'
];
foreach($fields as $field) {
// Process value of field if set in the page process.register_user
if (!isset($data[$field]) && $form->value($field)) {
$data[$field] = $form->value($field);
}
}
unset($data['password1']);
unset($data['password2']);
// Don't store the username: that is part of the filename
unset($data['username']);
2015-12-11 10:34:40 -07:00
// Extra lowercase to ensure file is saved lowercase
$username = strtolower($username);
$inflector = new Inflector();
$data['fullname'] = isset($data['fullname']) ? $data['fullname'] : $inflector->titleize($username);
$data['title'] = isset($data['title']) ? $data['title'] : 'Administrator';
2015-12-10 12:07:36 +01:00
$data['state'] = 'enabled';
$data['access'] = ['admin' => ['login' => true, 'super' => true], 'site' => ['login' => true]];
// Create user object and save it
$user = new User($data);
$file = CompiledYamlFile::instance($this->grav['locator']->findResource('user://accounts/' . $username . YAML_EXT, true, true));
$user->file($file);
$user->save();
$user = User::load($username);
//Login user
$this->grav['session']->user = $user;
unset($this->grav['user']);
$this->grav['user'] = $user;
$user->authenticated = $user->authorize('site.login');
$messages = $this->grav['messages'];
$messages->add($this->grav['language']->translate('PLUGIN_ADMIN.LOGIN_LOGGED_IN'), 'info');
$this->grav->redirect($this->base);
2015-12-10 12:07:36 +01:00
break;
}
}
/**
2015-07-30 12:20:25 +02:00
* If the admin plugin is set as active, initialize the admin
*/
public function onPluginsInitialized()
{
// Only activate admin if we're inside the admin path.
if ($this->active) {
2016-04-07 21:01:38 +02:00
// Have a unique Admin-only Cache key
if (method_exists($this->grav['cache'], 'setKey')){
$cache = $this->grav['cache'];
$cache_key = $cache->getKey();
$cache->setKey($cache_key . '$');
}
2016-04-07 21:01:38 +02:00
// Turn on Twig autoescaping
if (method_exists($this->grav['twig'], 'setAutoescape')) {
$this->grav['twig']->setAutoescape(true);
}
2016-04-07 21:01:38 +02:00
if (php_sapi_name() == 'cli-server') {
throw new \RuntimeException('The Admin Plugin cannot run on the PHP built-in webserver. It needs Apache, Nginx or another full-featured web server.', 500);
}
2015-10-20 10:36:48 +02:00
$this->grav['debugger']->addMessage("Admin Basic");
$this->initializeAdmin();
2014-09-09 14:03:01 -06:00
// Disable Asset pipelining (old method - remove this after Grav is updated)
if (!method_exists($this->grav['assets'],'setJsPipeline')) {
$this->config->set('system.assets.css_pipeline', false);
$this->config->set('system.assets.js_pipeline', false);
}
// Replace themes service with admin.
2016-01-28 14:38:01 +01:00
$this->grav['themes'] = function () {
require_once __DIR__ . '/classes/themes.php';
return new Themes($this->grav);
};
2014-08-05 13:06:38 -07:00
}
2014-09-09 14:03:01 -06:00
// We need popularity no matter what
require_once __DIR__ . '/classes/popularity.php';
$this->popularity = new Popularity();
2014-08-05 13:06:38 -07:00
}
protected function initializeController($task, $post) {
require_once __DIR__ . '/classes/controller.php';
$controller = new AdminController($this->grav, $this->template, $task, $this->route, $post);
$controller->execute();
$controller->redirect();
}
2014-08-05 13:06:38 -07:00
/**
* Sets longer path to the home page allowing us to have list of pages when we enter to pages section.
*/
public function onPagesInitialized()
2014-08-05 13:06:38 -07:00
{
$this->session = $this->grav['session'];
2014-08-05 13:06:38 -07:00
// Set original route for the home page.
$home = '/' . trim($this->config->get('system.home.alias'), '/');
2014-10-06 19:03:57 -06:00
// set the default if not set before
$this->session->expert = $this->session->expert ?: false;
2014-10-06 19:03:57 -06:00
// set session variable if it's passed via the url
if ($this->uri->param('mode') == 'expert') {
$this->session->expert = true;
} elseif ($this->uri->param('mode') == 'normal') {
$this->session->expert = false;
}
2014-08-05 13:06:38 -07:00
/** @var Pages $pages */
$pages = $this->grav['pages'];
$this->grav['admin']->routes = $pages->routes();
// Remove default route from routes.
if (isset($this->grav['admin']->routes['/'])) {
unset($this->grav['admin']->routes['/']);
}
$page = $pages->dispatch('/', true);
// If page is null, the default page does not exist, and we cannot route to it
if ($page) {
$page->route($home);
}
2014-08-05 13:06:38 -07:00
// Make local copy of POST.
$post = !empty($_POST) ? $_POST : array();
// Handle tasks.
2014-09-22 15:49:53 -06:00
$this->admin->task = $task = !empty($post['task']) ? $post['task'] : $this->uri->param('task');
2014-08-05 13:06:38 -07:00
if ($task) {
$this->initializeController($task, $post);
2014-08-05 13:06:38 -07:00
} elseif ($this->template == 'logs' && $this->route) {
// Display RAW error message.
echo $this->admin->logEntry();
exit();
}
$self = $this;
2014-08-05 13:06:38 -07:00
2015-12-09 21:37:34 -07:00
// make sure page is not frozen!
unset($this->grav['page']);
$this->admin->pagesCount();
// Replace page service with admin.
$this->grav['page'] = function () use ($self) {
$page = new Page;
// If the page cannot be found in other plugins, try looking in admin plugin itself.
if (file_exists(__DIR__ . "/pages/admin/{$self->template}.md")) {
$page->init(new \SplFileInfo(__DIR__ . "/pages/admin/{$self->template}.md"));
$page->slug(basename($self->template));
return $page;
}
// Allows pages added by plugins in admin
$plugins = $this->grav['plugins'];
$locator = $this->grav['locator'];
foreach($plugins as $plugin) {
$path = $locator->findResource(
"user://plugins/{$plugin->name}/admin/pages/{$self->template}.md");
2015-10-26 21:30:33 +01:00
if ($path) {
2015-10-26 21:30:33 +01:00
$page->init(new \SplFileInfo($path));
$page->slug(basename($self->template));
return $page;
}
}
2016-01-21 09:46:38 +02:00
return null;
};
if (empty($this->grav['page'])) {
if($this->session->user->count()){
$event = $this->grav->fireEvent('onPageNotFound');
if (isset($event->page)) {
unset($this->grav['page']);
$this->grav['page'] = $event->page;
} else {
throw new \RuntimeException('Page Not Found', 404);
}
}else{
$this->grav->redirect($this->base);
}
}
// Explicitly set a timestamp on assets
$this->grav['assets']->setTimestamp(substr(md5(GRAV_VERSION . $this->grav['config']->checksum()),0,10));
2014-08-05 13:06:38 -07:00
}
2016-01-10 17:17:04 +01:00
/**
* Handles initializing the assets
*/
public function onAssetsInitialized()
{
// Disable Asset pipelining
$assets = $this->grav['assets'];
$assets->setJsPipeline(false);
$assets->setCssPipeline(false);
}
2014-08-05 13:06:38 -07:00
/**
* Add twig paths to plugin templates.
*/
public function onTwigTemplatePaths()
2014-08-05 13:06:38 -07:00
{
$twig_paths = [];
$this->grav->fireEvent('onAdminTwigTemplatePaths', new Event(['paths' => &$twig_paths]));
$twig_paths[] = __DIR__ . '/themes/' . $this->theme . '/templates';
$this->grav['twig']->twig_paths = $twig_paths;
2014-08-05 13:06:38 -07:00
}
/**
* Set all twig variables for generating output.
*/
public function onTwigSiteVariables()
2014-08-05 13:06:38 -07:00
{
$twig = $this->grav['twig'];
2014-08-05 13:06:38 -07:00
$twig->twig_vars['location'] = $this->template;
$twig->twig_vars['base_url_relative_frontend'] = $twig->twig_vars['base_url_relative'] ?: '/';
$twig->twig_vars['admin_route'] = trim($this->config->get('plugins.admin.route'), '/');
$twig->twig_vars['base_url_relative'] =
$twig->twig_vars['base_url_simple'] . '/' . $twig->twig_vars['admin_route'];
2016-01-24 19:27:53 -07:00
$twig->twig_vars['theme_url'] = $this->grav['locator']->findResource('plugin://admin/themes/' . $this->theme, false);
2014-08-29 11:59:43 +03:00
$twig->twig_vars['base_url'] = $twig->twig_vars['base_url_relative'];
$twig->twig_vars['base_path'] = GRAV_ROOT;
2014-08-05 13:06:38 -07:00
$twig->twig_vars['admin'] = $this->admin;
// Gather Plugin-hooked nav items
2015-10-26 21:30:33 +01:00
$this->grav->fireEvent('onAdminMenu');
2014-08-05 13:06:38 -07:00
switch ($this->template) {
case 'dashboard':
$twig->twig_vars['popularity'] = $this->popularity;
// Gather Plugin-hooked dashboard items
$this->grav->fireEvent('onAdminDashboard');
break;
2014-08-05 13:06:38 -07:00
case 'pages':
$path = $this->route;
if (!$path) {
$path = '/';
}
if (!isset($this->pages[$path])) {
$page = null;
} else {
$page = $this->pages[$path];
}
if ($page != null) {
$twig->twig_vars['file'] = File::instance($page->filePath());
$twig->twig_vars['media_types'] = str_replace('defaults,', '',
implode(',.', array_keys($this->config->get('media'))));
}
2014-08-05 13:06:38 -07:00
break;
}
}
2014-09-03 22:22:03 -06:00
2016-01-10 17:17:04 +01:00
/**
* Handles the shutdown
*/
public function onShutdown()
{
2014-09-09 14:03:01 -06:00
// Just so we know that we're in this debug mode
2014-09-06 18:13:04 -06:00
if ($this->config->get('plugins.admin.popularity.enabled')) {
// Only track non-admin
if (!$this->active) {
$this->popularity->trackHit();
}
}
}
2015-07-30 12:20:25 +02:00
/**
* Handles getting GPM updates
*/
public function onTaskGPM()
{
2016-04-08 11:21:11 +02:00
$task = 'GPM';
if (!$this->admin->authorize(['admin.maintenance', 'admin.super'])) {
$this->admin->json_response = [
'status' => 'unauthorized',
'message' => $this->admin->translate('PLUGIN_ADMIN.INSUFFICIENT_PERMISSIONS_FOR_TASK') . ' ' . $task . '.'
];
return false;
}
$action = $_POST['action']; // getUpdatable | getUpdatablePlugins | getUpdatableThemes | gravUpdates
$flush = isset($_POST['flush']) && $_POST['flush'] == true ? true : false;
if (isset($this->grav['session'])) {
$this->grav['session']->close();
}
try {
$gpm = new GPM($flush);
switch ($action) {
case 'getUpdates':
$resources_updates = $gpm->getUpdatable();
if ($gpm->grav != null) {
$grav_updates = [
"isUpdatable" => $gpm->grav->isUpdatable(),
"assets" => $gpm->grav->getAssets(),
"version" => GRAV_VERSION,
"available" => $gpm->grav->getVersion(),
"date" => $gpm->grav->getDate(),
"isSymlink" => $gpm->grav->isSymlink()
];
echo json_encode([
"status" => "success",
"payload" => ["resources" => $resources_updates, "grav" => $grav_updates, "installed" => $gpm->countInstalled(), 'flushed' => $flush]
]);
} else {
echo json_encode(["status" => "error", "message" => "Cannot connect to the GPM"]);
}
break;
}
} catch (\Exception $e) {
echo json_encode(["status" => "error", "message" => $e->getMessage()]);
}
exit;
}
/**
* Get list of form field types specified in this plugin. Only special types needs to be listed.
*
* @return array
*/
public function getFormFieldTypes()
{
return [
'column' => [
2016-03-03 20:59:59 +02:00
'input@' => false
],
'columns' => [
2016-03-03 20:59:59 +02:00
'input@' => false
],
'fieldset' => [
2016-03-03 20:59:59 +02:00
'input@' => false
],
'section' => [
2016-03-03 20:59:59 +02:00
'input@' => false
],
'tab' => [
2016-03-03 20:59:59 +02:00
'input@' => false
],
'tabs' => [
2016-03-03 20:59:59 +02:00
'input@' => false
2016-03-10 19:56:38 +02:00
],
2016-03-15 14:38:17 +02:00
'key' => [
'input@' => false
],
2016-03-10 19:56:38 +02:00
'list' => [
'array' => true
]
];
}
2015-07-30 12:20:25 +02:00
/**
* Initialize the admin.
*
* @throws \RuntimeException
*/
2014-09-03 22:22:03 -06:00
protected function initializeAdmin()
{
$this->enable([
'onTwigExtensions' => ['onTwigExtensions', 1000],
'onPagesInitialized' => ['onPagesInitialized', 1000],
'onTwigTemplatePaths' => ['onTwigTemplatePaths', 1000],
'onTwigSiteVariables' => ['onTwigSiteVariables', 1000],
'onAssetsInitialized' => ['onAssetsInitialized', 1000],
'onTask.GPM' => ['onTaskGPM', 0]
]);
2014-09-03 22:22:03 -06:00
2015-12-10 12:08:06 -07:00
// Initialize admin class.
require_once __DIR__ . '/classes/admin.php';
2015-07-27 12:56:16 -06:00
// Check for required plugins
if (!$this->grav['config']->get('plugins.login.enabled') ||
!$this->grav['config']->get('plugins.form.enabled') ||
!$this->grav['config']->get('plugins.email.enabled')) {
throw new \RuntimeException('One of the required plugins is missing or not enabled');
}
// Initialize Admin Language if needed
/** @var Language $language */
$language = $this->grav['language'];
if ($language->enabled() && empty($this->grav['session']->admin_lang)) {
$this->grav['session']->admin_lang = $language->getLanguage();
}
// Decide admin template and route.
$path = trim(substr($this->uri->route(), strlen($this->base)), '/');
2015-12-10 11:02:19 -07:00
if (empty($this->template)) {
$this->template = 'dashboard';
}
// Can't access path directly...
if ($path && $path != 'register') {
$array = explode('/', $path, 2);
$this->template = array_shift($array);
$this->route = array_shift($array);
}
2014-09-03 22:22:03 -06:00
$this->admin = new Admin($this->grav, $this->base, $this->template, $this->route);
2016-02-04 20:59:08 +01:00
// And store the class into DI container.
$this->grav['admin'] = $this->admin;
// Double check we have system.yam, site.yaml etc
$config_path = $this->grav['locator']->findResource('user://config');
foreach ($this->admin->configurations() as $config_file) {
$config_file = "{$config_path}/{$config_file}.yaml";
if (!file_exists($config_file)) {
touch($config_file);
}
}
// Get theme for admin
$this->theme = $this->config->get('plugins.admin.theme', 'grav');
$assets = $this->grav['assets'];
2016-02-01 11:27:34 -08:00
$translations = 'this.GravAdmin = this.GravAdmin || {}; if (!this.GravAdmin.translations) this.GravAdmin.translations = {}; ' . PHP_EOL . 'this.GravAdmin.translations.PLUGIN_ADMIN = {';
// Enable language translations
$translations_actual_state = $this->config->get('system.languages.translations');
$this->config->set('system.languages.translations', true);
$strings = ['EVERYTHING_UP_TO_DATE',
'UPDATES_ARE_AVAILABLE',
'IS_AVAILABLE_FOR_UPDATE',
'AND',
'IS_NOW_AVAILABLE',
'CURRENT',
'UPDATE_GRAV_NOW',
'TASK_COMPLETED',
'UPDATE',
'UPDATING_PLEASE_WAIT',
'GRAV_SYMBOLICALLY_LINKED',
'OF_YOUR',
'OF_THIS',
'HAVE_AN_UPDATE_AVAILABLE',
'UPDATE_AVAILABLE',
'UPDATES_AVAILABLE',
'FULLY_UPDATED',
2015-10-21 19:54:10 +02:00
'DAYS',
'PAGE_MODES',
'PAGE_TYPES',
2016-02-01 11:27:34 -08:00
'ACCESS_LEVELS',
2016-02-01 16:34:50 -08:00
'NOTHING_TO_SAVE',
'FILE_UNSUPPORTED',
'FILE_ERROR_ADD',
'FILE_ERROR_UPLOAD'
2015-10-21 19:54:10 +02:00
];
foreach($strings as $string) {
2016-02-01 11:27:34 -08:00
$separator = (end($strings) === $string) ? '' : ',';
$translations .= '"' . $string .'": "' . $this->admin->translate('PLUGIN_ADMIN.' . $string) . '"' . $separator;
}
2016-02-01 11:27:34 -08:00
$translations .= '};';
// set the actual translations state back
$this->config->set('system.languages.translations', $translations_actual_state);
$assets->addInlineJs($translations);
2014-09-03 22:22:03 -06:00
}
/**
2016-01-10 17:17:04 +01:00
* Add the Admin Twig Extensions
*/
public function onTwigExtensions()
{
require_once(__DIR__.'/twig/AdminTwigExtension.php');
$this->grav['twig']->twig->addExtension(new AdminTwigExtension());
}
2015-12-10 12:08:06 -07:00
2016-01-10 17:17:04 +01:00
/**
* Check if the current route is under the admin path
*
* @return bool
*/
public function isAdminPath()
{
if ($this->uri->route() == $this->base ||
substr($this->uri->route(), 0, strlen($this->base) + 1) == $this->base . '/') {
return true;
}
return false;
}
public function onAdminDashboard()
{
$this->grav['twig']->plugins_hooked_dashboard_widgets_top[] = ['template' => 'dashboard-maintenance'];
$this->grav['twig']->plugins_hooked_dashboard_widgets_top[] = ['template' => 'dashboard-statistics'];
$this->grav['twig']->plugins_hooked_dashboard_widgets_main[] = ['template' => 'dashboard-pages'];
}
2014-08-05 13:06:38 -07:00
}