Serverless PHP websites
In this guide, you will learn how to set up assets for your serverless PHP website.
This guide assumes that you have already gotten started with Bref. If you haven't, get started first.
Architecture
Websites usually contain 2 parts:
- PHP code, running on AWS Lambda + API Gateway (read the HTTP applications guide)
- static assets (CSS, JSâŠ), hosted on AWS S3 (opens in a new tab)
To combine both, we can use AWS CloudFront (opens in a new tab). CloudFront acts both as a CDN and as reverse proxy to route requests to PHP or assets on S3.
This lets us host everything under the same domain and support both HTTP and HTTPS.
If you don't want to use Cloudfront, you can read the older version of this documentation (opens in a new tab) which featured running PHP and the assets on two different domains.
Setup
While it is possible to set up CloudFront manually, the easiest approach is to use the Server-side website construct of the Lift plugin (opens in a new tab).
First install the plugin:
serverless plugin install -n serverless-liftThen add this configuration to serverless.yml:
# ...
 
plugins:
    - ./vendor/bref/bref
    - serverless-lift
 
constructs:
    website:
        type: server-side-website
        assets:
            '/build/*': public/build
            '/vendor/*': public/vendor
            '/favicon.ico': public/favicon.ico
            '/robots.txt': public/robots.txt
            # add here any file or directory that needs to be served from S3Before deploying, compile your assets:
npm run prodNow deploy everything:
serverless deployLift will create all the required resources and take care of uploading your assets to S3 automatically. You can access your website using the URL that Lift outputs at the end the deployment.
The first deployment takes 5 minutes because CloudFront is a distributed service. The next deployments that do not modify CloudFront's configuration will not suffer from this delay.
Assets in templates
Assets referenced in Blade templates should be via the asset() helper:
<script src="{{ asset('js/app.js') }}"></script>If your templates reference some assets via direct path, you should edit them to use the asset() helper:
- <img src="/images/logo.png"/>
+ <img src="{{ asset('images/logo.png') }}"/>Custom domain name
When using CloudFront, the custom domain must be set up on CloudFront, not API Gateway. If you have already set up your domain on API Gateway you will need to remove it before continuing.
The first thing to do is register the domain in ACM (AWS Certificate Manager) to get an HTTPS certificate. This step is not optional.
- Open this link (opens in a new tab) or manually go in the ACM Console and click "Request a new certificate" in the us-east-1region (CloudFront requires certificates fromus-east-1because it is a global service)
- Add your domain name and click "Next".
- Choose the domain validation of your choice:
- domain validation will require you to create DNS entries (this is recommended because it renews the certificate automatically)
- email validation will require you to click a link you will receive in an email sent to admin@your-domain.com
 
Copy the ARN of the ACM certificate. It should look like this:
arn:aws:acm:us-east-1:216536346254:certificate/322f12ee-1165-4bfa-a41f-08c932a2935dNext, add your domain name and certificate in serverless.yml:
# ...
 
constructs:
    website:
        # ...
        domain: mywebsite.com
        certificate: <certificate ARN>The last step will be to point your domain name DNS records to the CloudFront domain:
- copy the domain outputted by Lift during serverless deploy(or runserverless infoto retrieve it)
- create a CNAME to point your domain name to this URL
- if you use Route53 you can read the official guide (opens in a new tab)
- if you use another registrar and you want to point your root domain (without www.) to CloudFront, you will need to use a registrar that supports this (for example CloudFlare allows this with a technique called CNAME flattening (opens in a new tab))
 
Lift supports more advanced use cases like multiple domains, root domain to www redirects, and more. Check out the Lift documentation (opens in a new tab).
Compressing HTTP responses
By default, Lift enables gzip and brotli compression for static assets served from S3. That means that if a client supports compressed responses (via the Accept-Encoding header), CloudFront will cache and serve a compressed version of the asset (opens in a new tab), which is usually smaller and faster to transfer.
However, dynamic responses generated by PHP (for example HTML pages) are not compressed. The reason is that CloudFront's cache is disabled for requests to PHP (Lift sets up the CachingDisabled policy (opens in a new tab)), and therefore CloudFront does not compress non-cached responses.
You can compress PHP responses by using an HTTP middleware that compresses the response body and sets the Content-Encoding header. Here is an example for Laravel:
class CompressResponse
{
    public function handle(Request $request, Closure $next)
    {
        /** @var \Illuminate\Http\JsonResponse $response */
        $response = $next($request);
 
        if (! in_array('gzip', $request->getEncodings())) {
            return $response;
        }
        if ($response->headers->has('Content-Encoding')) {
            return $response;
        }
 
        $content = $response->getContent();
        if (! $content) {
            return $response;
        }
 
        $response->setContent(gzencode($content, 9));
        $response->headers->set('Content-Encoding', 'gzip');
        $response->headers->set('Content-Length', (string) strlen($content));
 
        return $response;
    }
}