tkotosz / fn-fdk-php
PHP Function Development Kit for Fn
Requires
- php: >=7.1
- react/http: ^0.8.3
- react/socket: ^1.1
This package is auto-updated.
Last update: 2025-01-08 07:44:42 UTC
README
fn-fdk-go provides convenience functions for writing php fn code
Installing fn-fdk-php
You can install it manually with composer:
composer require tkotosz/fn-fdk-php
Or just use the php init image to create new funtion like this:
fn init --init-image tkotosz/fn-php-init myfunc
This will generate the necessary files for your function including the composer json and docker file to install this fdk.
Creating a PHP Function
Writing a PHP function is simply a matter of writing a handler function that you pass to the FDK to invoke each time your function is called.
Start by creating a php function with fn init
:
fn init --init-image tkotosz/fn-php-init phpfunc
cd phpfunc
This creates a simple hello world function in func.php
:
<?php require('vendor/autoload.php'); $fdk = new Tkotosz\FnPhpFdk\Fdk(); $fdk->handle(function ($input) { $name = 'World'; if (isset($input['name'])) { $name = $input['name']; } return ['message' => 'Hello ' . $name]; });
The handler function takes the input that is sent to the function and returns a response. By default the input is treated as json which automatically converted to an array and the response is also an array (or json serializable object) which automatically converted to json. Using the FDK you don't have to worry about reading the http request or sending back the response. The FDK let's you focus on your function logic and not the mechanics.
Now run it!
fn deploy --local --app fdkdemo fn invoke fdkdemo phpfunc
You should see the following output:
{"message":"Hello World"}
Run it with input:
echo -n '{"name":"Tibor"}' | fn invoke fdkdemo phpfunc
You should see the following output:
{"message":"Hello Tibor"}
Now you have a basic running php function that you can modify and add what you want.
Function Context
Function invocation context details are available through an optional function argument.
To receive a context object, simply add a second argument to your handler function.
In the following example the callId
is obtained from the context and included in
the response message:
<?php require('vendor/autoload.php'); $fdk = new Tkotosz\FnPhpFdk\Fdk(); $fdk->handle(function ($input, $ctx) { $name = 'World'; if (isset($input['name'])) { $name = $input['name']; } return ['message' => 'Hello ' . $name, 'callId' => $ctx->getCallId()]; });
Run it:
echo -n '{"name":"Tibor"}' | fn invoke fdkdemo phpfunc
You should see a similar output:
{"message":"Hello Tibor","callId":"01D0F7QX2QNG8G00GZJ00001YV"}
The context contains other context information about the request such as:
ctx->getConfig
: An object containing function config variables (from the environment variables)ctx->getHeaders
: An object containing input headers for the event as lists of stringsctx->getDeadline
: ADateTimeImmutable
object indicating when the function call must be processed byctx->getCallId
: The call ID of the current callctx->getId
: The function ID of the current functionctx->getMemory
: Amount of ram in MB allocated to this functionctx->getContentType
: The incoming request content type (if set, otherwise null)ctx->setResponseHeader(key,values...)
: Sets a response header to one or more valuesctx->addResponseHeader(key,values...)
: Appends values to an existing response headerctx->responseContentType
: Sets the response content type of the functionctx->setResponseStatus
: Sets the response status code of the function (default: 200)
Handling input/output
By default the FDK will try to json decode the input, likewise by default the output of a function will be treated as a JSON object and converted using json_encode()
.
To change the handling of the input you can add an additional options
parameter to fdk->handle
that specifies the input handling strategy:
$fdk->handle(function ($input) use ($fdk) { return ['message' => 'Hello ' . ($input ?: 'World')]; }, ['inputMode' => 'string']);
valid input modes are:
json
(the default) attempts to parse the input as jsonstring
always treats input as a stringstream
passes the input stream (streaming request body) to your function
To change the output handling of your function from the default you should wrap the result value using a response decorator:
$fdk->handle(function ($input) use ($fdk) { return $fdk->rawResult('Hello '. ($input ?: 'World')); }, ['inputMode' => 'string']);
the available decorators are:
rawResult({string|ReadableStreamInterface})
passes the result directly to the response - the value can be a string or a readable streamstreamResult({resource|ReadableStreamInterface})
pipes the contents of theresource
orReadableStreamInterface
into the output - this allows processing of data from files or HTTP responses
Using HTTP headers and setting HTTP status codes
You can read http headers passed into a function invocation using $ctx->getHeaderValue($key)
, this returns the first header value of the header matching key
or you can use $ctx->getHeaders()
or $ctx->getHeaderValues($key)
methods as well.
$fdk->handle(function ($input, $context) use ($fdk) { return $context->getHeaders(); // this will return all request headers as json }, ['inputMode' => 'string']);
Outbound headers and the HTTP status code can be modified in a similar way:
$fdk->handle(function ($input, $context) { $context->setResponseStatus(201); $context->setResponseContentType('text/plain'); $context->setResponseHeader('X-Awesomeness-level', 100); $context->addResponseHeader('X-Awesome-number', 1); $context->addResponseHeader('X-Awesome-number', 2); return 'Hello '. ($input ?: 'World'); }, ['inputMode' => 'string']);
Examples
See examples here.