'XBackBone', 'base_url' => isset($_SERVER['HTTPS']) ? 'https://' . $_SERVER['HTTP_HOST'] : 'http://' . $_SERVER['HTTP_HOST'], 'storage_dir' => 'storage', 'displayErrorDetails' => false, 'db' => [ 'connection' => 'sqlite', 'dsn' => __DIR__ . '/../resources/database/xbackbone.db', 'username' => null, 'password' => null, ], ], require __DIR__ . '/../config.php'); if (!$config['displayErrorDetails']) { $config['routerCacheFile'] = __DIR__ . '/../resources/cache/routes.cache.php'; } $container = new Container(['settings' => $config]); $container['logger'] = function ($container) { $logger = new Logger('app'); $streamHandler = new RotatingFileHandler(__DIR__ . '/../logs/log.txt', 10, Logger::DEBUG); $streamHandler->setFormatter(new LineFormatter("[%datetime%] %channel%.%level_name%: %message% %context% %extra%\n", "Y-m-d H:i:s", true)); $logger->pushHandler($streamHandler); return $logger; }; // Session init Session::init('xbackbone_session', __DIR__ . '/../resources/sessions'); // Set the database dsn $dsn = $config['db']['connection'] === 'sqlite' ? __DIR__ . '/../' . $config['db']['dsn'] : $config['db']['dsn']; DB::setDsn($config['db']['connection'] . ':' . $dsn, $config['db']['username'], $config['db']['password']); $container['database'] = function ($container) use (&$config) { return DB::getInstance(); }; Lang::build(substr(@$_SERVER['HTTP_ACCEPT_LANGUAGE'], 0, 2), __DIR__. '/../resources/lang/'); $container['view'] = function ($container) use (&$config) { $view = new \Slim\Views\Twig(__DIR__ . '/../resources/templates', [ 'cache' => __DIR__ . '/../resources/cache', 'autoescape' => 'html', 'debug' => $config['displayErrorDetails'], 'auto_reload' => $config['displayErrorDetails'], ]); // Instantiate and add Slim specific extension $router = $container->get('router'); $uri = \Slim\Http\Uri::createFromEnvironment(new \Slim\Http\Environment($_SERVER)); $view->addExtension(new Slim\Views\TwigExtension($router, $uri)); $view->getEnvironment()->addGlobal('config', $config); $view->getEnvironment()->addGlobal('request', $container->get('request')); $view->getEnvironment()->addGlobal('alerts', Session::getAlert()); $view->getEnvironment()->addGlobal('session', Session::all()); $view->getEnvironment()->addGlobal('current_lang', Lang::getInstance()->getLang()); $view->getEnvironment()->addGlobal('PLATFORM_VERSION', PLATFORM_VERSION); $view->getEnvironment()->addFunction(new Twig_Function('route', 'route')); $view->getEnvironment()->addFunction(new Twig_Function('lang', 'lang')); $view->getEnvironment()->addFunction(new Twig_Function('urlFor', 'urlFor')); return $view; }; $container['errorHandler'] = function ($container) { return function (\Slim\Http\Request $request, \Slim\Http\Response $response, $exception) use (&$container) { if ($exception instanceof \App\Exceptions\UnauthorizedException) { return $container->view->render($response->withStatus(403), 'errors/403.twig'); } $container->logger->critical('Fatal error during app execution', [$exception, $exception->getTraceAsString()]); return $container->view->render($response->withStatus(500), 'errors/500.twig', ['exception' => $exception]); }; }; $container['notFoundHandler'] = function ($container) { return function (\Slim\Http\Request $request, \Slim\Http\Response $response) use (&$container) { $response->withStatus(404)->withHeader('Content-Type', 'text/html'); return $container->view->render($response, 'errors/404.twig'); }; }; $app = new App($container); // Permanently redirect paths with a trailing slash to their non-trailing counterpart $app->add(function (\Slim\Http\Request $request, \Slim\Http\Response $response, callable $next) { $uri = $request->getUri(); $path = $uri->getPath(); if ($path !== '/' && substr($path, -1) === '/') { $uri = $uri->withPath(substr($path, 0, -1)); if ($request->getMethod() === 'GET') { return $response->withRedirect((string)$uri, 301); } else { return $next($request->withUri($uri), $response); } } return $next($request, $response); }); // Load the application routes require 'app/routes.php';