Move application code to app folder

This commit is contained in:
Frederic Guillot 2016-08-18 21:02:49 -04:00
parent 32b4f684d0
commit 1f88eb50f3
No known key found for this signature in database
GPG Key ID: 92D77191BA7FBC99
100 changed files with 314 additions and 406 deletions

View File

@ -3,3 +3,5 @@
.gitignore
.dockerignore
.travis.yml
tests
scripts

1
.gitattributes vendored
View File

@ -29,5 +29,4 @@ vendor/bin/ export-ignore
vendor/fguillot/picofeed/picofeed export-ignore
# unittests
phpunit.xml export-ignore
**/tests/ export-ignore

43
.gitignore vendored
View File

@ -1,47 +1,13 @@
# Compiled source #
###################
*.com
*.class
*.dll
*.exe
*.o
*.so
*.pyc
# Packages #
############
# it's better to unpack these files and commit the raw source
# git has its own built in compression methods
*.7z
*.dmg
*.gz
*.iso
*.jar
*.rar
*.tar
*.zip
# Logs and databases #
######################
*.log
*.sql
*.sqlite
*.sqlite-journal
# OS generated files #
######################
.DS_Store
ehthumbs.db
Icon?
Thumbs.db
*.swp
.*.swp
*~
*.lock
*.out
# IDE generated files #
######################
.idea
.buildpath
.project
@ -50,12 +16,9 @@ Thumbs.db
*.sublime-workspace
.nbproject
nbproject
# App specific #
################
config.php
!models/*
!controllers/*
!app/models/*
!app/controllers/*
!app/templates/*
rules/*.php
data/favicons/*.*
/nbproject/private/

View File

@ -74,5 +74,5 @@ if (! is_writable(FAVICON_DIRECTORY)) {
// Include password_compat for PHP < 5.5
if (version_compare(PHP_VERSION, '5.5.0', '<')) {
require __DIR__.'/lib/password.php';
require __DIR__.'/libraries/password.php';
}

37
app/common.php Normal file
View File

@ -0,0 +1,37 @@
<?php
require __DIR__.'/../vendor/autoload.php';
if (file_exists(__DIR__.'/../config.php')) {
require __DIR__.'/../config.php';
}
require __DIR__.'/constants.php';
require __DIR__.'/check_setup.php';
require __DIR__.'/functions.php';
PicoDb\Database::setInstance('db', function() {
$db = new PicoDb\Database(array(
'driver' => 'sqlite',
'filename' => Model\Database\get_path(),
));
if ($db->schema()->check(Schema\VERSION)) {
return $db;
}
else {
$errors = $db->getLogMessages();
$html = 'Unable to migrate the database schema, <strong>please copy and paste this message and create a bug report:</strong><hr/>';
$html .= '<pre><code>';
$html .= (isset($errors[0]) ? $errors[0] : 'Unknown SQL error').PHP_EOL.PHP_EOL;
$html .= '- PHP version: '.phpversion().PHP_EOL;
$html .= '- SAPI: '.php_sapi_name().PHP_EOL;
$html .= '- PDO Sqlite version: '.phpversion('pdo_sqlite').PHP_EOL;
$html .= '- Sqlite version: '.$db->getDriver()->getDatabaseVersion().PHP_EOL;
$html .= '- OS: '.php_uname();
$html .= '</code></pre>';
die($html);
}
});

View File

@ -1,18 +1,12 @@
<?php
require __DIR__.'/vendor/autoload.php';
if (file_exists(__DIR__.'/config.php')) {
require __DIR__.'/config.php';
}
defined('APP_VERSION') or define('APP_VERSION', Helper\parse_app_version('$Format:%d$','$Format:%H$'));
defined('HTTP_TIMEOUT') or define('HTTP_TIMEOUT', 20);
defined('HTTP_MAX_RESPONSE_SIZE') or define('HTTP_MAX_RESPONSE_SIZE', 2097152);
defined('BASE_URL_DIRECTORY') or define('BASE_URL_DIRECTORY', dirname($_SERVER['PHP_SELF']));
defined('ROOT_DIRECTORY') or define('ROOT_DIRECTORY', __DIR__);
defined('ROOT_DIRECTORY') or define('ROOT_DIRECTORY', implode(DIRECTORY_SEPARATOR, array(__DIR__, '..')));
defined('DATA_DIRECTORY') or define('DATA_DIRECTORY', ROOT_DIRECTORY.DIRECTORY_SEPARATOR.'data');
defined('FAVICON_DIRECTORY') or define('FAVICON_DIRECTORY', DATA_DIRECTORY.DIRECTORY_SEPARATOR.'favicons');
@ -45,32 +39,3 @@ defined('ENABLE_HSTS') or define('ENABLE_HSTS', true);
defined('BEANSTALKD_HOST') or define('BEANSTALKD_HOST', '127.0.0.1');
defined('BEANSTALKD_QUEUE') or define('BEANSTALKD_QUEUE', 'feeds');
defined('BEANSTALKD_TTL') or define('BEANSTALKD_TTL', 120);
require __DIR__.'/check_setup.php';
PicoDb\Database::setInstance('db', function() {
$db = new PicoDb\Database(array(
'driver' => 'sqlite',
'filename' => Model\Database\get_path(),
));
if ($db->schema()->check(Schema\VERSION)) {
return $db;
}
else {
$errors = $db->getLogMessages();
$html = 'Unable to migrate the database schema, <strong>please copy and paste this message and create a bug report:</strong><hr/>';
$html .= '<pre><code>';
$html .= (isset($errors[0]) ? $errors[0] : 'Unknown SQL error').PHP_EOL.PHP_EOL;
$html .= '- PHP version: '.phpversion().PHP_EOL;
$html .= '- SAPI: '.php_sapi_name().PHP_EOL;
$html .= '- PDO Sqlite version: '.phpversion('pdo_sqlite').PHP_EOL;
$html .= '- Sqlite version: '.$db->getDriver()->getDatabaseVersion().PHP_EOL;
$html .= '- OS: '.php_uname();
$html .= '</code></pre>';
die($html);
}
});

View File

@ -7,26 +7,22 @@ function get_server_variable($variable)
return isset($_SERVER[$variable]) ? $_SERVER[$variable] : '';
}
function param($name, $default_value = null)
{
return isset($_GET[$name]) ? $_GET[$name] : $default_value;
}
function int_param($name, $default_value = 0)
{
return isset($_GET[$name]) && ctype_digit($_GET[$name]) ? (int) $_GET[$name] : $default_value;
}
function value($name)
{
$values = values();
return isset($values[$name]) ? $values[$name] : null;
}
function values()
{
if (! empty($_POST)) {
@ -42,13 +38,11 @@ function values()
return array();
}
function body()
{
return file_get_contents('php://input');
}
function file_content($field)
{
if (isset($_FILES[$field])) {
@ -58,25 +52,21 @@ function file_content($field)
return '';
}
function uri()
{
return $_SERVER['REQUEST_URI'];
}
function is_post()
{
return $_SERVER['REQUEST_METHOD'] === 'POST';
}
function get_user_agent()
{
return get_server_variable('HTTP_USER_AGENT') ?: t('Unknown');
return get_server_variable('HTTP_USER_AGENT') ?: 'Unknown';
}
function get_ip_address()
{
$keys = array(
@ -99,5 +89,5 @@ function get_ip_address()
}
}
return t('Unknown');
return 'Unknown';
}

View File

@ -7,7 +7,6 @@ function force_download($filename)
header('Content-Disposition: attachment; filename="'.$filename.'"');
}
function status($status_code)
{
$sapi_name = php_sapi_name();
@ -19,14 +18,12 @@ function status($status_code)
}
}
function redirect($url, $status_code = 302)
{
header('Location: '.$url, true, $status_code);
exit;
}
function json(array $data, $status_code = 200)
{
status($status_code);
@ -37,7 +34,6 @@ function json(array $data, $status_code = 200)
exit;
}
function text($data, $status_code = 200)
{
status($status_code);
@ -48,7 +44,6 @@ function text($data, $status_code = 200)
exit;
}
function html($data, $status_code = 200)
{
status($status_code);
@ -59,7 +54,6 @@ function html($data, $status_code = 200)
exit;
}
function xml($data, $status_code = 200)
{
status($status_code);
@ -70,7 +64,6 @@ function xml($data, $status_code = 200)
exit;
}
function raw($data, $status_code = 200)
{
status($status_code);
@ -78,7 +71,6 @@ function raw($data, $status_code = 200)
exit;
}
function binary($data, $status_code = 200)
{
status($status_code);
@ -90,7 +82,6 @@ function binary($data, $status_code = 200)
exit;
}
function csp(array $policies = array())
{
$policies['default-src'] = "'self'";
@ -115,25 +106,21 @@ function csp(array $policies = array())
header('Content-Security-Policy: '.$values);
}
function nosniff()
{
header('X-Content-Type-Options: nosniff');
}
function xss()
{
header('X-XSS-Protection: 1; mode=block');
}
function hsts()
{
header('Strict-Transport-Security: max-age=31536000');
}
function xframe($mode = 'DENY', array $urls = array())
{
header('X-Frame-Options: '.$mode.' '.implode(' ', $urls));

View File

@ -2,10 +2,11 @@
namespace Router;
// Load controllers: bootstrap('controllers', 'controller1', 'controller2')
use Closure;
function bootstrap()
{
$files = \func_get_args();
$files = func_get_args();
$base_path = array_shift($files);
foreach ($files as $file) {
@ -38,7 +39,7 @@ function before_action($name, $value = null)
}
// Execute an action
function action($name, \Closure $callback)
function action($name, Closure $callback)
{
$handler = isset($_GET['action']) ? $_GET['action'] : 'default';
@ -50,7 +51,7 @@ function action($name, \Closure $callback)
}
// Execute an action only for POST requests
function post_action($name, \Closure $callback)
function post_action($name, Closure $callback)
{
if ($_SERVER['REQUEST_METHOD'] === 'POST') {
action($name, $callback);
@ -58,7 +59,7 @@ function post_action($name, \Closure $callback)
}
// Execute an action only for GET requests
function get_action($name, \Closure $callback)
function get_action($name, Closure $callback)
{
if ($_SERVER['REQUEST_METHOD'] === 'GET') {
action($name, $callback);
@ -66,7 +67,7 @@ function get_action($name, \Closure $callback)
}
// Run when no action have been executed before
function notfound(\Closure $callback)
function notfound(Closure $callback)
{
before('notfound');
before_action('notfound');

View File

@ -4,7 +4,6 @@ namespace Session;
const SESSION_LIFETIME = 2678400;
function open($base_path = '/', $save_path = '', $session_lifetime = SESSION_LIFETIME)
{
if ($save_path !== '') {
@ -40,19 +39,16 @@ function open($base_path = '/', $save_path = '', $session_lifetime = SESSION_LIF
}
}
function close()
{
session_destroy();
}
function flash($message)
{
$_SESSION['flash_message'] = $message;
}
function flash_error($message)
{
$_SESSION['flash_error_message'] = $message;

View File

@ -2,7 +2,7 @@
namespace Template;
const PATH = 'templates/';
const PATH = __DIR__.'/../templates/';
// Template\load('template_name', ['bla' => 'value']);
function load()
@ -28,7 +28,6 @@ function load()
return ob_get_clean();
}
function layout($template_name, array $template_args = array(), $layout_name = 'layout')
{
return load($layout_name, $template_args + array('content_for_layout' => load($template_name, $template_args)));

143
app/core/Translator.php Normal file
View File

@ -0,0 +1,143 @@
<?php
namespace Translator;
const PATH = __DIR__.'/../locales/';
function translate($identifier)
{
$args = func_get_args();
array_shift($args);
array_unshift($args, get($identifier, $identifier, $args));
foreach ($args as &$arg) {
$arg = htmlspecialchars($arg, ENT_QUOTES, 'UTF-8', false);
}
return \call_user_func_array(
'sprintf',
$args
);
}
function translate_no_escaping($identifier)
{
$args = func_get_args();
array_shift($args);
array_unshift($args, get($identifier, $identifier));
return call_user_func_array(
'sprintf',
$args
);
}
function number($number)
{
return number_format(
$number,
get('number.decimals', 2),
get('number.decimals_separator', '.'),
get('number.thousands_separator', ',')
);
}
function currency($amount)
{
$position = get('currency.position', 'before');
$symbol = get('currency.symbol', '$');
$str = '';
if ($position === 'before') {
$str .= $symbol;
}
$str .= number($amount);
if ($position === 'after') {
$str .= ' '.$symbol;
}
return $str;
}
function datetime($format, $timestamp)
{
if (strtoupper(substr(PHP_OS, 0, 3)) == 'WIN') {
$format = preg_replace('#(?<!%)((?:%%)*)%e#', '\1%#d', $format);
$format = preg_replace('#(?<!%)((?:%%)*)%k#', '\1%#H', $format);
}
return strftime($format, (int) $timestamp);
}
function get($identifier, $default = '', array $values = array())
{
$locales = container();
$translation = $default;
if (isset($locales[$identifier])) {
if (is_array($locales[$identifier])) {
$translation = plural($identifier, $default, $values);
} else {
$translation = $locales[$identifier];
}
}
return $translation;
}
function plural($identifier, $default, array $values)
{
$locales = container();
$plural = 0;
foreach ($values as $value) {
if (is_numeric($value)) {
$value = abs($value);
$plural = (int) $locales['plural']($value);
break;
}
}
for ($i = $plural; $i >= 0; --$i) {
if (isset($locales[$identifier][$i])) {
return $locales[$identifier][$i];
}
}
return $default;
}
function load($language)
{
setlocale(LC_TIME, $language.'.UTF-8');
$path = PATH.$language;
$locales = array();
if (is_dir($path)) {
$dir = new \DirectoryIterator($path);
foreach ($dir as $fileinfo) {
if (strpos($fileinfo->getFilename(), '.php') !== false) {
$locales = array_merge($locales, include $fileinfo->getPathname());
}
}
}
container($locales);
}
function container($locales = null)
{
static $values = array();
if ($locales !== null) {
$values = $locales;
}
return $values;
}

26
app/functions.php Normal file
View File

@ -0,0 +1,26 @@
<?php
function tne()
{
return call_user_func_array('\Translator\translate_no_escaping', func_get_args());
}
function t()
{
return call_user_func_array('\Translator\translate', func_get_args());
}
function c()
{
return call_user_func_array('\Translator\currency', func_get_args());
}
function n()
{
return call_user_func_array('\Translator\number', func_get_args());
}
function dt()
{
return call_user_func_array('\Translator\datetime', func_get_args());
}

View File

@ -245,6 +245,4 @@ return array(
// 'Item title links to' => '',
// 'Original' => '',
// 'Last login:' => '',
// 'Search' => '',
// 'There are no results for your search' => '',
);

View File

@ -245,6 +245,4 @@ return array(
'Item title links to' => 'Titulek článku odkazuje na',
'Original' => 'Originál',
'Last login:' => 'Poslední přihlášení:',
'Search' => 'Hledat',
'There are no results for your search' => 'Žádné výsledky vašeho hledání',
);

View File

@ -245,6 +245,4 @@ return array(
// 'Item title links to' => '',
// 'Original' => '',
// 'Last login:' => '',
// 'Search' => '',
// 'There are no results for your search' => '',
);

View File

@ -245,6 +245,4 @@ return array(
// 'Item title links to' => '',
// 'Original' => '',
// 'Last login:' => '',
// 'Search' => '',
// 'There are no results for your search' => '',
);

View File

@ -245,6 +245,4 @@ return array(
// 'Item title links to' => '',
// 'Original' => '',
// 'Last login:' => '',
// 'Search' => '',
// 'There are no results for your search' => '',
);

View File

@ -247,6 +247,4 @@ return array(
'Item title links to' => 'アイテムのタイトルのリンク先',
'Original' => '元のページ',
// 'Last login:' => '',
// 'Search' => '',
// 'There are no results for your search' => '',
);

View File

@ -245,6 +245,4 @@ return array(
// 'Item title links to' => '',
// 'Original' => '',
// 'Last login:' => '',
// 'Search' => '',
// 'There are no results for your search' => '',
);

View File

@ -245,6 +245,4 @@ return array(
'Item title links to' => 'Заголовок статьи ведет на',
'Original' => 'Оригинал',
'Last login:' => 'Последний вход:',
'Search' => 'Поиск',
'There are no results for your search' => 'По вашему запросу ничего не нашлось',
);

View File

@ -245,6 +245,4 @@ return array(
// 'Item title links to' => '',
// 'Original' => '',
// 'Last login:' => '',
// 'Search' => '',
// 'There are no results for your search' => '',
);

View File

@ -245,6 +245,4 @@ return array(
// 'Item title links to' => '',
// 'Original' => '',
// 'Last login:' => '',
// 'Search' => '',
// 'There are no results for your search' => '',
);

View File

@ -245,6 +245,4 @@ return array(
'Item title links to' => 'Öğe başlığıyla ilişkili bağlantılar',
'Original' => 'Orijinal',
'Last login:' => 'En son sisteme giriş zamanı:',
// 'Search' => '',
// 'There are no results for your search' => '',
);

View File

@ -245,6 +245,4 @@ return array(
// 'Item title links to' => '',
// 'Original' => '',
// 'Last login:' => '',
// 'Search' => '',
// 'There are no results for your search' => '',
);

View File

@ -4,7 +4,6 @@ namespace Schema;
use PDO;
use Helper;
use Model\Config;
const VERSION = 44;

View File

@ -25,29 +25,29 @@
},
"autoload": {
"files": [
"helpers/app.php",
"helpers/csrf.php",
"helpers/favicon.php",
"helpers/form.php",
"helpers/template.php",
"lib/Translator.php",
"lib/Request.php",
"lib/Response.php",
"lib/Router.php",
"lib/Session.php",
"lib/Template.php",
"models/config.php",
"models/service.php",
"models/user.php",
"models/feed.php",
"models/item.php",
"models/proxy.php",
"models/schema.php",
"models/auto_update.php",
"models/database.php",
"models/remember_me.php",
"models/group.php",
"models/favicon.php"
"app/schemas/sqlite.php",
"app/helpers/app.php",
"app/helpers/csrf.php",
"app/helpers/favicon.php",
"app/helpers/form.php",
"app/helpers/template.php",
"app/core/translator.php",
"app/core/request.php",
"app/core/response.php",
"app/core/router.php",
"app/core/session.php",
"app/core/template.php",
"app/models/config.php",
"app/models/service.php",
"app/models/user.php",
"app/models/feed.php",
"app/models/item.php",
"app/models/proxy.php",
"app/models/auto_update.php",
"app/models/database.php",
"app/models/remember_me.php",
"app/models/group.php",
"app/models/favicon.php"
],
"classmap": [
"vendor/fguillot/json-rpc/src/",

View File

@ -1,6 +1,6 @@
<?php
require '../common.php';
require __DIR__.'/../app/common.php';
use Model\Feed;
use Model\Group;

View File

@ -1,10 +1,20 @@
<?php
require __DIR__.'/common.php';
require __DIR__.'/app/common.php';
Router\bootstrap(__DIR__.'/controllers', 'common', 'console', 'user', 'config', 'item', 'history', 'bookmark', 'feed', 'search');
Router\bootstrap(
__DIR__.'/app/controllers',
'common',
'console',
'user',
'config',
'item',
'history',
'bookmark',
'feed',
'search'
);
// Page not found
Router\notfound(function() {
Response\redirect('?action=unread');
});

View File

@ -1,6 +1,6 @@
<?php
require __DIR__.'/common.php';
require __DIR__.'/app/common.php';
use JsonRPC\Server;
use Model\Config;

View File

@ -1 +0,0 @@
Deny from all

View File

@ -1,180 +0,0 @@
<?php
namespace Translator {
const PATH = 'locales/';
function translate($identifier)
{
$args = \func_get_args();
\array_shift($args);
\array_unshift($args, get($identifier, $identifier, $args));
foreach ($args as &$arg) {
$arg = htmlspecialchars($arg, ENT_QUOTES, 'UTF-8', false);
}
return \call_user_func_array(
'sprintf',
$args
);
}
function translate_no_escaping($identifier)
{
$args = \func_get_args();
\array_shift($args);
\array_unshift($args, get($identifier, $identifier));
return \call_user_func_array(
'sprintf',
$args
);
}
function number($number)
{
return number_format(
$number,
get('number.decimals', 2),
get('number.decimals_separator', '.'),
get('number.thousands_separator', ',')
);
}
function currency($amount)
{
$position = get('currency.position', 'before');
$symbol = get('currency.symbol', '$');
$str = '';
if ($position === 'before') {
$str .= $symbol;
}
$str .= number($amount);
if ($position === 'after') {
$str .= ' '.$symbol;
}
return $str;
}
function datetime($format, $timestamp)
{
if (strtoupper(substr(PHP_OS, 0, 3)) == 'WIN') {
$format = preg_replace('#(?<!%)((?:%%)*)%e#', '\1%#d', $format);
$format = preg_replace('#(?<!%)((?:%%)*)%k#', '\1%#H', $format);
}
return strftime($format, (int) $timestamp);
}
function get($identifier, $default = '', array $values = array())
{
$locales = container();
$translation = $default;
if (isset($locales[$identifier])) {
if (is_array($locales[$identifier])) {
$translation = plural($identifier, $default, $values);
} else {
$translation = $locales[$identifier];
}
}
return $translation;
}
function plural($identifier, $default, array $values)
{
$locales = container();
$plural = 0;
foreach ($values as $value) {
if (is_numeric($value)) {
$value = abs($value);
$plural = (int) $locales['plural']($value);
break;
}
}
for ($i = $plural; $i >= 0; --$i) {
if (isset($locales[$identifier][$i])) {
return $locales[$identifier][$i];
}
}
return $default;
}
function load($language)
{
setlocale(LC_TIME, $language.'.UTF-8');
$path = PATH.$language;
$locales = array();
if (is_dir($path)) {
$dir = new \DirectoryIterator($path);
foreach ($dir as $fileinfo) {
if (strpos($fileinfo->getFilename(), '.php') !== false) {
$locales = array_merge($locales, include $fileinfo->getPathname());
}
}
}
container($locales);
}
function container($locales = null)
{
static $values = array();
if ($locales !== null) {
$values = $locales;
}
return $values;
}
}
namespace {
function tne()
{
return call_user_func_array('\Translator\translate_no_escaping', func_get_args());
}
function t()
{
return call_user_func_array('\Translator\translate', func_get_args());
}
function c()
{
return call_user_func_array('\Translator\currency', func_get_args());
}
function n()
{
return call_user_func_array('\Translator\number', func_get_args());
}
function dt()
{
return call_user_func_array('\Translator\datetime', func_get_args());
}
}

View File

@ -1 +0,0 @@
Deny from all

View File

@ -2,7 +2,7 @@
use Pheanstalk\Pheanstalk;
require __DIR__.'/common.php';
require __DIR__.'/app/common.php';
if (php_sapi_name() !== 'cli') {
die('This script can run only from the command line.'.PHP_EOL);

View File

@ -3,7 +3,7 @@
REF_LANG=${1:-fr_FR}
###
APP_DIR=`dirname $0`/../
APP_DIR=`dirname $0`/../app
LANG_FILE=$APP_DIR/locales/$REF_LANG/translations.php
TMPFILE=`mktemp`
@ -20,4 +20,4 @@ do
done < $TMPFILE
# delete the work file
rm $TMPFILE
rm $TMPFILE

View File

@ -2,7 +2,7 @@
<?php
$reference_lang = 'fr_FR';
$reference_file = 'locales/'.$reference_lang.'/translations.php';
$reference_file = 'app/locales/'.$reference_lang.'/translations.php';
$reference = include $reference_file;
@ -43,11 +43,9 @@ function update_missing_locales(array $reference, $outdated_file)
}
foreach (new DirectoryIterator('locales') as $fileInfo) {
foreach (new DirectoryIterator('app/locales') as $fileInfo) {
if (! $fileInfo->isDot() && $fileInfo->isDir() && $fileInfo->getFilename() !== $reference_lang && $fileInfo->getFilename() !== 'en_US') {
$filename = 'locales/'.$fileInfo->getFilename().'/translations.php';
$filename = 'app/locales/'.$fileInfo->getFilename().'/translations.php';