-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathindex.php
218 lines (177 loc) · 7.4 KB
/
index.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
<?php
require_once __DIR__ . '/func.php';
use Symfony\Component\Translation\Translator;
use Symfony\Component\Translation\Loader\YamlFileLoader;
use Twig\Loader\FilesystemLoader as TwigFSLoader;
use Twig\Environment as TwigEnvironment;
use PhpProxyHunter\TranslationExtension;
ini_set('memory_limit', '512M');
// Capture the full request URI
$requestUri = $_SERVER['REQUEST_URI'] ?? '';
// Extract the path part of the URI
$requestPath = parse_url($requestUri, PHP_URL_PATH);
// Check if the request is for a static file
$staticFileExtensions = ['css', 'js', 'png', 'jpg', 'jpeg', 'gif', 'ico', 'svg', 'woff', 'woff2', 'ttf', 'eot'];
$fileExtension = pathinfo($requestPath, PATHINFO_EXTENSION);
if (in_array($fileExtension, $staticFileExtensions)) {
$filePath = __DIR__ . $requestPath; // Construct the full file path
if (file_exists($filePath)) {
// Serve the file content directly
header('Content-Type: ' . mime_content_type($filePath));
read_file($filePath);
exit; // Stop further execution
}
// If file doesn't exist, respond with 404
http_response_code(404);
echo "$filePath file not found.";
exit;
}
// Remove leading and trailing slashes and split into segments
$pathSegments = explode('/', trim($requestPath, '/'));
// Default values for controller and action
$controllerName = !empty($pathSegments[0]) ? ucfirst($pathSegments[0]) . 'Controller' : 'DefaultController';
$actionName = !empty($pathSegments[1]) ? $pathSegments[1] . 'Action' : 'indexAction';
// Any remaining segments are treated as parameters
$params = array_slice($pathSegments, 2);
// Autoload controllers
function autoloadController($className)
{
$filePath = __DIR__ . '/controllers/' . $className . '.php';
if (file_exists($filePath)) {
require_once $filePath;
}
}
spl_autoload_register('autoloadController');
// Initialize Twig
$loader = new TwigFSLoader(__DIR__ . '/views');
$twig = new TwigEnvironment($loader);
// Initialize Translator
$translator = new Translator('en');
$translator->addLoader('yaml', new YamlFileLoader());
// Load translation files for different languages
$translator->addResource('yaml', __DIR__ . '/translations/messages.en.yaml', 'en');
$translator->addResource('yaml', __DIR__ . '/translations/messages.id.yaml', 'id');
// Detect locale from URL query parameter or cookie
$locale = isset($_GET['hl']) ? $_GET['hl'] : (isset($_COOKIE['locale']) ? $_COOKIE['locale'] : 'en');
// Validate the locale to ensure it's one of the supported locales
$validLocales = ['en', 'id']; // Add more locales as needed
if (!in_array($locale, $validLocales)) {
$locale = 'en'; // Default locale
}
// Set the detected locale
$translator->setLocale($locale);
$translator->setFallbackLocales(['en']);
// Get all PHP files in the 'translations/ext' folder
$extensionsPath = __DIR__ . '/translations/ext';
$files = glob($extensionsPath . '/*.php');
foreach ($files as $file) {
require_once $file; // Include the file (make sure the extension class is autoloaded)
// Get the class name (assuming class name matches the file name)
$className = "\PhpProxyHunter\\" . basename($file, '.php');
if (!class_exists($className)) {
$className = basename($file, '.php');
}
// If the class exists and implements Twig\Extension\ExtensionInterface
if (class_exists($className) && is_subclass_of($className, \Twig\Extension\ExtensionInterface::class)) {
$extension = null;
if (strpos($className, 'TranslationExtension') !== false && isset($translator)) {
// Pass the $translator to the TranslationExtension constructor
$extension = new $className($translator);
} else {
// No special arguments needed for other extensions
$extension = new $className();
}
// Add the extension to Twig
if ($extension) {
$twig->addExtension($extension);
}
}
}
// Dispatcher
try {
// Load when controller exists
if (class_exists($controllerName)) {
// Instantiate the controller
$controllerInstance = new $controllerName();
// Check if the action method exists
if (!method_exists($controllerInstance, $actionName)) {
throw new Exception("Action $actionName not found in $controllerName.");
}
// Call the action with parameters
call_user_func_array([$controllerInstance, $actionName], $params);
}
// Extract the base name for the view
$baseControllerName = strtolower(str_replace('Controller', '', $controllerName));
$baseActionName = strtolower(str_replace('Action', '', $actionName));
// Determine the view to render based on controller and action
$viewName = $baseControllerName == 'default' ? $baseActionName : "{$baseControllerName}/{$baseActionName}";
// Check if the view exists in the 'views/' folder
$rawViewPath = __DIR__ . "/views/{$viewName}.twig";
$viewPath = realpath($rawViewPath);
if (!$viewPath) {
if (is_debug()) {
throw new Exception("View file for **$viewName** not found. (**$rawViewPath**)");
}
throw new Exception("View file for **$viewName** not found.");
}
// Render the view
$render_html = $twig->render("{$viewName}.twig", [
'date' => date(DATE_RFC3339),
'debug' => is_debug(),
'locale' => $locale
]);
} catch (Exception $e) {
// Handle errors by rendering a Twig error page
http_response_code(404);
$md = new Parsedown();
$render_html = $twig->render('404.twig', [
'errorMessage' => $md->text($e->getMessage()),
'errorFile' => $e->getFile(),
'errorLine' => $e->getLine(),
'debug' => is_debug(),
'trace' => print_r($e->getTrace(), true),
'locale' => $locale
]);
}
/**
* Adds the "nofollow" attribute to external HTTP(S) links in the provided HTML string,
* except for whitelisted domains.
*
* This function parses the provided HTML, finds all anchor tags (`<a>`), and checks whether
* the `href` attribute contains an external HTTP(S) URL. If the URL is external and not in the
* whitelist, the function adds the `rel="nofollow noopener noreferer"` and `target="_blank"`
* attributes to the anchor tag.
*
* @param string $html The input HTML string containing anchor (`<a>`) tags.
* @param array $whitelist An array of domains that should be excluded from the nofollow processing.
*
* @return string The modified HTML string with `nofollow`, `noopener`, `noreferrer`, and `target="_blank"` added to external HTTP(S) links, except for those in the whitelist.
*/
function addNoFollowToExternalLinks($html, $whitelist = [])
{
// Create a DOMDocument instance
$dom = new DOMDocument();
// Suppress warnings due to malformed HTML
@$dom->loadHTML($html);
// Find all anchor tags
$links = $dom->getElementsByTagName('a');
// Regex to match HTTP(S) URLs
$pattern = '/^https?:\/\/[^\s\/$.?#].[^\s]*$/';
foreach ($links as $link) {
// Get the href attribute
$url = $link->getAttribute('href');
// Validate the URL is an HTTP(S) URL using regex
if (preg_match($pattern, $url)) {
// Check if the URL is external and not in the whitelist
$host = parse_url($url, PHP_URL_HOST);
if (strpos($url, $_SERVER['HTTP_HOST']) === false && !in_array($host, $whitelist)) {
// Add rel="nofollow noopener noreferer" and target="_blank" if external and not whitelisted
$link->setAttribute('rel', 'nofollow noopener noreferer');
$link->setAttribute('target', '_blank');
}
}
}
// Save and return the modified HTML
return $dom->saveHTML();
}
echo addNoFollowToExternalLinks($render_html, ['www.webmanajemen.com', 'webmanajemen.com']);