websharks / php-web-server
PHP Web Server by WebSharks, Inc.
Requires
- php-64bit: >=5.4
- ext-mbstring: *
This package is not auto-updated.
Last update: 2025-01-16 01:30:06 UTC
README
Warning: PHP's built-in web server was designed to aid application development. It may also be useful for testing purposes or for application demonstrations that are run in controlled environments. It is not intended to be a full-featured web server. It should not be used on a public network. See this article for more info.
Installation
Via Composer
Requires PHP 5.4+ w/ mbstring
extension. Known to work with recent versions of PHP 7.x also.
{ "require": { "websharks/php-web-server": "dev-master" } }
Or, via Git
$ mkdir -p ~/projects/websharks; $ git clone https://github.com/websharks/php-web-server ~/projects/websharks/php-web-server;
Running the Web Server
Required Directory Structure
~/www
(absolute root base directory; passed as-t
below)/localhost
(document root sub-folder matching host name below)index.php
... and any other web-accessible files can live here.
- ... for as many host names as you like.
$ mkdir -p ~/www/localhost; $ echo 'It works!' > ~/www/localhost/index.php;
Start Server from a Terminal Window
$ php \ -S localhost:8080 \ -t ~/www \ -d variables_order=EGPCS \ ~/projects/websharks/php-web-server/src/includes/router.php
Note: It is normal for the process to remain open (i.e., occupy) your terminal session. While the server is running, you can open a browser and navigate your web-accessible files. Press Ctrl-C in your terminal window to stop the web server when you're done testing.
Note: The document root starts out with just the ~/www
base, and then automatically becomes: ~/www/localhost
when you visit: http://localhost:8080/
in a browser. In other words, the document root changes based on the host name in the request, making it possible to run one PHP Web Server against multiple host names w/ multiple document roots that are satisfied dynamically by router.php
.
Additional Information
WordPress-Compatible
The router.php
file is capable of automatically falling back on a root index.php
file in your document root sub-folder for each host name (i.e., it performs a mod_rewrite
simulation). This makes the PHP Web Server compatible with applications such as WordPress also.
Limitation: Standard WordPress only. Not compatible with WordPress Multisite Networking.
Using a Custom Host Name
If you've edited your /etc/hosts
file and would like to use a custom host name, that's fine. Simply start the web server with a different host name, and create a document root sub-folder matching that host name.
Why sudo
here? Most systems require sudo
to bind to ports lower than 1024
. sudo
is required in this case, because we want to use the default standard port 80
so we can access the web-accessible files by typing a custom host name: http://example.dev
(without the port number). Thus, binding to the default port 80
requires sudo
.
Note: This will only work on port 80
if it is available. For instance, if you're also running Apache, Nginx, or another web server that consumes port 80
, you must stop that server to free-up port 80
for PHP.
$ sudo php \ -S example.dev:80 \ -t ~/www \ -d variables_order=EGPCS \ ~/projects/websharks/php-web-server/src/includes/router.php
Now, if your /etc/hosts
are configured properly, you can access:
http://example.dev/
Document root starts as ~/www
and becomes: ~/www/example.dev
when you visit http://example.dev/
One Document Root w/ Multiple Sub-Domains
You can force the document root sub-folder to a fixed location by hard-coding the full document root as follows. Note the addition of /example.dev
in the -t
argument below.
Why sudo
here? Most systems require sudo
to bind to ports lower than 1024
. sudo
is required in this case, because we want to use the default standard port 80
so we can access the web-accessible files by typing a custom host name: http://example.dev
(without the port number). Thus, binding to the default port 80
requires sudo
.
Note: This will only work on port 80
if it is available. For instance, if you're also running Apache, Nginx, or another web server that consumes port 80
, you must stop that server to free-up port 80
for PHP.
$ sudo php \ -S example.dev:80 \ -t ~/www/example.dev \ -d variables_order=EGPCS \ ~/projects/websharks/php-web-server/src/includes/router.php
Now, if your /etc/hosts
are configured properly, you can access:
http://example.dev/
http://sub1.example.dev/
http://sub2.example.dev/
And, the document root is now (always) ~/www/example.dev
so long as your hard-coded document root sub-folder ends w/ the requested root host name. In this case: example.dev
The only downside to doing it this way is that you lose the ability to run a single PHP Web Server process that will be capable of locating a nested document root sub-folder based on the host name. In this example, you're hard-coding the document root sub-folder, which limits the PHP Web Server to that document root only; i.e., to a single root host name.
Environment Variables
The standard PHP environment variables are available in the $_SERVER
super-global as expected. The WebSharks PHP Web Router will also add some additional environment variables for convenience, and in an effort to fill-in common FastCGI Params that would ordinarily be provided by Nginx or another PHP-FPM integration.
Here is an example dump of $_SERVER
that will give you a quick glimpse. This was produced at the URL: http://localhost:8080/index.php/path/info?v=1
(with PATH_INFO
to show a more complete example).
[HTTP_HOST] => localhost:8080 [DOCUMENT_BASE] => /Users/websharks/www [DOCUMENT_ROOT] => /Users/websharks/www/localhost [REMOTE_ADDR] => ::1 [REMOTE_PORT] => 61573 [SERVER_PORT] => 8080 [SERVER_NAME] => localhost [SERVER_ADDR] => 127.0.0.1 [SERVER_PROTOCOL] => HTTP/1.1 [SERVER_SOFTWARE] => PHP 7.0.4 Development Server [REQUEST_METHOD] => GET [REQUEST_SCHEME] => http [REDIRECT_STATUS] => 200 [QUERY_STRING] => v=1 [PATH_INFO] => /path/info [SCRIPT_NAME] => /index.php [DOCUMENT_URI] => /index.php [PHP_SELF] => /index.php/path/info [REQUEST_URI] => /index.php/path/info?v=1 [SCRIPT_FILENAME] => /Users/websharks/www/localhost/index.php [PATH_TRANSLATED] => /Users/websharks/www/localhost/index.php [REQUEST_TIME_FLOAT] => 1457530515.8927 [REQUEST_TIME] => 1457530515
In addition to $_SERVER
, there is the $_ENV
super-global, which contains local environment variables from the shell you use to start the PHP Web Server. By default, PHP will automatically exclude the $_ENV
super-global in the context of the built-in PHP Web Server. However, it is suggested that you include $_ENV
, as it provides a lot of useful information—such as USER
, HOME
, SHELL
, etc. This is accomplished with the -d variables_order=EGPCS
flag (note the E
inclusion) when starting the server.
Whenever $_ENV
is filled (recommended), the WebSharks Web Router will create an internal copy of all environment variables. It merges $_SERVER
into $_ENV
($_SERVER
has the highest precedence); and then it will repopulate $_SERVER
with the full set of environment variables so they can all be accessed from the $_SERVER
super-global. This is how most FastCGI implementations work also. It improves compatibility and offers some added convenience.
MIME Types Library
The following file extensions are automatically served with these content-type headers.
// Text files.
'md' => 'text/plain; charset=utf-8',
'txt' => 'text/plain; charset=utf-8',
// Log files.
'log' => 'text/plain; charset=utf-8',
// Translation files.
'mo' => 'application/x-gettext-translation',
'po' => 'text/x-gettext-translation; charset=utf-8',
'pot' => 'text/x-gettext-translation; charset=utf-8',
// SQL files.
'sql' => 'text/plain; charset=utf-8',
'sqlite' => 'text/plain; charset=utf-8',
// Template files.
'tmpl' => 'text/plain; charset=utf-8',
'tpl' => 'text/plain; charset=utf-8',
// Server config files.
'admins' => 'text/plain; charset=utf-8',
'cfg' => 'text/plain; charset=utf-8',
'conf' => 'text/plain; charset=utf-8',
'htaccess' => 'text/plain; charset=utf-8',
'htaccess-apache' => 'text/plain; charset=utf-8',
'htpasswd' => 'text/plain; charset=utf-8',
'ini' => 'text/plain; charset=utf-8',
// CSS/JavaScript files.
'css' => 'text/css; charset=utf-8',
'js' => 'application/x-javascript; charset=utf-8',
'json' => 'application/json; charset=utf-8',
// PHP scripts/files.
'php' => 'text/html; charset=utf-8',
'phps' => 'text/html; charset=utf-8',
// ASP scripts/files.
'asp' => 'text/html; charset=utf-8',
'aspx' => 'text/html; charset=utf-8',
// Perl scripts/files.
'cgi' => 'text/html; charset=utf-8',
'pl' => 'text/html; charset=utf-8',
// HTML/XML files.
'dtd' => 'application/xml-dtd; charset=utf-8',
'hta' => 'application/hta; charset=utf-8',
'htc' => 'text/x-component; charset=utf-8',
'htm' => 'text/html; charset=utf-8',
'html' => 'text/html; charset=utf-8',
'shtml' => 'text/html; charset=utf-8',
'xhtml' => 'application/xhtml+xml; charset=utf-8',
'xml' => 'text/xml; charset=utf-8',
'xsl' => 'application/xslt+xml; charset=utf-8',
'xslt' => 'application/xslt+xml; charset=utf-8',
'xsd' => 'application/xsd+xml; charset=utf-8',
// Document files.
'csv' => 'text/csv; charset=utf-8',
'doc' => 'application/msword',
'docx' => 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
'odt' => 'application/vnd.oasis.opendocument.text',
'pdf' => 'application/pdf',
'rtf' => 'application/rtf',
'xls' => 'application/vnd.ms-excel',
// Image/animation files.
'ai' => 'image/vnd.adobe.illustrator',
'blend' => 'application/x-blender',
'bmp' => 'image/bmp',
'eps' => 'image/eps',
'fla' => 'application/vnd.adobe.flash',
'gif' => 'image/gif',
'ico' => 'image/x-icon',
'jpe' => 'image/jpeg',
'jpeg' => 'image/jpeg',
'jpg' => 'image/jpeg',
'png' => 'image/png',
'psd' => 'image/vnd.adobe.photoshop',
'pspimage' => 'image/vnd.corel.psp',
'svg' => 'image/svg+xml',
'swf' => 'application/x-shockwave-flash',
'tif' => 'image/tiff',
'tiff' => 'image/tiff',
// Audio files.
'mid' => 'audio/midi',
'midi' => 'audio/midi',
'mp3' => 'audio/mp3',
'wav' => 'audio/wav',
'wma' => 'audio/x-ms-wma',
// Video files.
'avi' => 'video/avi',
'flv' => 'video/x-flv',
'ogg' => 'video/ogg',
'ogv' => 'video/ogg',
'mp4' => 'video/mp4',
'mov' => 'movie/quicktime',
'mpg' => 'video/mpeg',
'mpeg' => 'video/mpeg',
'qt' => 'video/quicktime',
'webm' => 'video/webm',
'wmv' => 'audio/x-ms-wmv',
// Font files.
'eot' => 'application/vnd.ms-fontobject',
'otf' => 'application/x-font-otf',
'ttf' => 'application/x-font-ttf',
'woff' => 'application/x-font-woff',
'woff2' => 'application/x-font-woff',
// Archive files.
'7z' => 'application/x-7z-compressed',
'dmg' => 'application/x-apple-diskimage',
'gtar' => 'application/x-gtar',
'gz' => 'application/gzip',
'iso' => 'application/iso-image',
'jar' => 'application/java-archive',
'phar' => 'application/php-archive',
'rar' => 'application/x-rar-compressed',
'tar' => 'application/x-tar',
'tgz' => 'application/x-gtar',
'zip' => 'application/zip',
// Other misc files.
'bat' => 'application/octet-stream',
'bin' => 'application/octet-stream',
'class' => 'application/octet-stream',
'com' => 'application/octet-stream',
'dll' => 'application/octet-stream',
'exe' => 'application/octet-stream',
'sh' => 'application/octet-stream',
'bash' => 'application/octet-stream',
'zsh' => 'application/octet-stream',
'so' => 'application/octet-stream',
Caveats
-
https://
is completely unsupported at this time. -
HTTP 2 is completely unsupported at this time (HTTP 1.1 only).
-
PHP's web server runs only one single-threaded process. PHP scripts will stall if a request is blocked; i.e., requests are processed synchronously. One at a time (max). This means, for instance, that you can't have a PHP script that attempts to connect to another PHP script that is also served by the same PHP Web Server. That results in a timeout every time.
For instance, if
http://localhost:8080/script1.php
contains:<?php file_get_contents('http://localhost:8080/script2.php'); // `script2.php` will not load because `script1.php` is blocking it. // In other words, `script2.php` cannot run until `script1.php` finishes. // It can't finish, because it is stuck waiting for `script2.php`.
In short,
script2.php
will never run, andscript1.php
will timeout waiting for it.