copyright | lastupdated | keywords | subcollection | ||
---|---|---|---|---|---|
|
2024-10-18 |
code engine, functions, stateless code snippet, code snippet, stateless |
codeengine |
{{site.data.keyword.attribute-definition-list}}
{: #fun-exchanging-data}
Invoking a {{site.data.keyword.codeengineshort}} functions service (or simply a function) is accomplished on the client side through HTTP requests. Information can be passed to the function in the following ways:
- As part of the URL (path and optionally, query parameters)
- Through HTTP request headers
- Through the HTTP request body
A caller receives these values on the client side as an HTTP response. The response data passes as follows:
- As HTTP response code
- Through HTTP response headers
- Through the HTTP response body
All rules and capabilities to invoke a function through HTTP requests and the resulting HTTP responses are the external data interface.
An incoming HTTP request is forwarded to the function code for processing by using the rules of the internal request data interface. The results that are provided by the function code must follow the rules of the internal response data interface.
The following diagram shows information flow from a client that uses the external data interface. The process uses the {{site.data.keyword.codeengineshort}} functions service and the internal request data interface to the function code. Likewise, the resulting data flows from the function code to the internal response data interface through the {{site.data.keyword.codeengineshort}} functions service, and the external data interface flows back to the caller.
{: caption="Information flow for {{site.data.keyword.codeengineshort}} functions" caption-side="bottom"}
{: #fun-external-data-interface-request}
The external data interface definition is derived from standardized HTTP.
{: #fun-external-data-interface-MIME-types}
MIME types are used to define the format of the HTTP request information and the HTTP response.
The following IANA MIME types are supported for the content-type (Content-Type
) of the HTTP requests or HTTP responses. MIME type parameters are supported, but not included in this list:
- JSON type:
application/json; <parameters>
- Binary types:
audio/*; <parameters>
application/octet-stream; <parameters>
example/*; <parameters>
font/*; <parameters>
image/*; <parameters>
model/*; <parameters>
multipart/*; <parameters>
video/*; <parameters>
- Text types:
text/plain; <parameters>
text/html; <parameters>
text/*; <parameters>
- Percent-encoded type:
application/x-www-form-urlencoded; <parameters>
{: #fun-external-data-interface-request-data}
When a function is invoked, it can receive arbitrary data (request payload) in text or binary form. The presence of a Content-Type
value in the HTTP request header determines the form and encoding of the data that is sent to the function. The data structure, the format, and the content type must match. The {{site.data.keyword.codeengineshort}} functions service runs a minimal validation check on the data by considering the content-type and responds with the HTTP status code 400
in certain cases; for example, if the JSON data format or encoding is invalid.
For all other selected Content-Type
values, it is expected that the data format matches the restrictions of the MIME-type.
{: #fun-external-data-interface-providing-request-data=as=query}
Request data can be provided as key-value pairs in URL-encoded format (also called percent-encoded) in the HTTP URL.
Percent-encoding is widely used in web technology. All 8-bit characters are written as hexadecimal values preceded by a percentage sign (%). This type of encoding is also known as URL-encoding.
Example of invoking a function by using query parameters:
curl -v "https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud/?planet1=Mars&planet2=Jupiter"
{: screen}
{: #fun-external-data-interface-providing-request-data-in-body}
Request data is provided in the body section of the HTTP request. The format of the data must match the provided Content-Type
value.
Examples of invoking a function by using body data on UNIX®:
curl -v -H "Content-Type: application/x-www-form-urlencoded" -d 'location=planet%20earth' https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud
curl -v -H "Content-Type: application/json" -d "{'planet1': 'Mars', 'planet2': 'Jupiter'}" https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud
{: screen}
Example of invoking a function by using body data in Windows™:
curl -v -H "Content-Type: application/json" -d "{\"key_1\":\"Mars\",\"planet2\":\"Jupiter\"}" "https://function-nodejs-95.1057yuwab63w.us-east.codeengine.appdomain.cloud"
{: screen}
If the Content-Type
value is missing for a request, {{site.data.keyword.codeengineshort}} handles the data payload as described for application/json
.
{: tip}
{: #fun-external-data-interface-providing-request-header}
Request data is provided in key-value pairs in the header section of the HTTP request.
Example of invoking a function by using header fields:
curl -v -H "Sample_Data: Sample_Value" https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud
{: screen}
{: #fun-external-data-interface-providing-request-mixed-data}
{{site.data.keyword.codeengineshort}} functions support the use of different ways to provide request data within a single HTTP request.
Example of invoking a function by using body data and header fields:
curl -v -H "Sample_Data: Sample_Value" -H "Content-Type: application/x-www-form-urlencoded" -d 'planet1=Mars&planet2=Jupiter' https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud
{: screen}
Example of invoking a function by using body data, headers, and query parameters:
curl -v -H "Sample_Data: Sample_Value" -H "Content-Type: application/x-www-form-urlencoded" -d 'planet1=Mars&planet2=Jupiter' https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud/?planet1=Mars&planet2=Jupiter
{: screen}
{: #fun-internal-data-interface-requests}
The internal request data interface describes how the {{site.data.keyword.codeengineshort}} functions service delivers request data that is received on the external data interface to the function code. Independent of the coding language, you can implement function logic to put all request parameters in the main() function as the first parameter.
You can use any name for this parameter; in the proceeding examples, the parameter is called args
.
{: tip}
Also, the function code receives a set of automatically injected pre-defined environment variables.
Python example:
import os
def main(args):
# raw_input_data = args.__ce_body; # uncomment this line if client is providing data in the request body
all_input=args;
env=dict(os.environ)
print("Return body:",all_input)
return {
"statusCode": 200,
"body" : all_input,
}
{: screen}
Node.js example:
function main(args) {
const body = {
args,
env: process.env,
};
console.log(`Return body: ${JSON.stringify(body, null, 2)}`);
return {
statusCode: 200,
"body" : body,
};
}
module.exports.main = main;
{: screen}
The {{site.data.keyword.codeengineshort}} functions service uses different encodings for the input data to help ensure that the function code can access the input data in the args
argument. The encodings help ensure that structured data, plain text, data, and binary data can be part of the input data.
{: #fun-internal-data-interface-requests-text}
Text encoding is used for elements in the input args
parameter that contain text payload. The encoding is necessary to help ensure that special characters are escaped and do not damage the data structure of the args
parameter.
{{site.data.keyword.codeengineshort}} runs the text content encoding before it calls the main() function. The function's program logic might need to unescape them to re-create the original data.
Supported escape characters:
- Backspace is replaced with
\b
. - Form feed is replaced with
\f
. - Newline is replaced with
\n
. - Carriage return is replaced with
\r
. - Tab is replaced with
\t
. - Double quotation marks are replaced with
\"
. - Backslash is replaced with
\\
.
{: #fun-internal-data-interface-requests-Base64}
Base64 encoding is used for elements in the input args
parameter that can break the internal data structure of the args
parameter. Encoding it as Base64 helps ensure that the whole value consists only of readable characters in a 64-character alphabet.
For example:
"eyAicGxhbmV0MSI6Ik1hcnMiLCAicGxhbmV0MiI6Ikp1cGl0ZXIiIH0="
{: screen}
{: #fun-internal-data-interface-requests-percent}
Percent-encoding is widely used in web technology. All 8-bit characters are written as hexadecimal value preceded by a percentage sign (%). This type of encoding is also known as URL-encoding. The function receives the form-urlencoded
type data in raw format in the __ce_body
parameter.
The {{site.data.keyword.codeengineshort}} functions service passes the query parameters as-is (the leading ?
is not part of it) into the __ce_query
field of the args
parameter. This way, the function code receives this parameter in URL-encoded format; for example:
"x%5cb=1%22f4%20and%20"
{: screen}
{: #fun-args}
The args
parameter is a language-specific data structure (JSON object for Node.js and dict
(dictionary) type for Python) filled with the request input data in the following format:
Top-level elements | Description |
---|---|
__ce_<service_variable> |
{{site.data.keyword.codeengineshort}} service internal input argument. |
__ce_headers |
A copy of the HTTP request header element. |
<property> (optional) |
Top-level properties available if the HTTP request input data is provided in structured format (only if a content-type of application/json is used). |
{: caption="Elements for the args parameter"} |
{
"args": {
"__ce_body": "eyAicGxhbmV0MSI6Ik1hcnMiLCAicGxhbmV0MiI6Ikp1cGl0ZXIiIH0=",
"__ce_headers": {
"Accept": "*/*",
"User-Agent": "curl/7.58.0",
"X-Request-Id": "d03a1af0-bfc8-4a50-be7d-a72040c02cc9",
"Content-Type": "application/json",
},
"__ce_method": "GET",
"__ce_path": "/",
"__ce_query": "",
"planet1": "Mars",
"planet2": "Jupiter"
}
}
{: screen}
{: #fun-args-internal}
The {{site.data.keyword.codeengineshort}} functions service injects reserved input arguments to the args
input data.
The following parameters can appear in the args
argument depending how the caller provides the request data:
Parameter name | Value type | Description |
---|---|---|
__ce_method |
String | The HTTP request method (either a GET or POST request). |
__ce_headers |
Map of key-value pairs | Key-value pairs of HTTP request header. |
__ce_path |
String | URL path of the incoming HTTP request. |
__ce_body |
String | The request body entity, as a Base64-encoded string when request content type is binary, or as a plain string, otherwise. |
__ce_query |
String | The query parameters from the request as an unparsed string. The __ce_query parameter is a single string that contains the query parameters that are parsed from the URL, without the leading question mark (?), and separated by an ampersand (&). |
{: caption="Parameters for a {{site.data.keyword.codeengineshort}} function's input arguments"} |
The internal __ce_*
parameters cannot be overwritten with request data that is provided in the HTTP request. HTTP calls trying to overwrite fails with status code 400
(Bad Request).
{: #fun-args-query=parameters}
A function receives query parameter data as key-value pairs in the __ce_query
parameter, and is percent-encoded (URL-encoded). Also, each single query parameter is a key-value pair within the decoded format:
args field |
Is this set? | Description |
---|---|---|
__ce_body |
No | Body |
__ce_headers |
Yes | Header with no content-type |
__ce_query |
Yes | Query parameter (URL-encoded) |
Top-level property | Yes | Key-value pair (URL-decoded and text-encoded) |
{: caption="args parameter from query parameters when invoking functions"} |
Python example of accessing query parameters:
import os
def main(args):
query_parm_1 = args.get("key", "default_value") # get the value of one query parameter
try:
query = args["__ce_query"] # get all query parms
except:
query= "not part of request"
return {
"statusCode": 200,
"body": query,
}
{: screen}
Node.js example of accessing query parameters:
function main(args) {
var query_parm_1 = args.key // get the value of one query parameter
var query = args.__ce_query // get all query parameters
return {
statusCode: 200,
"body" : query,
};
}
module.exports.main = main;
{: screen}
{: #fun-args-heder-data}
A function receives request header data as key-value pairs in the __ce_headers
parameter. The key-value pair is converted into a canonical format, regardless of which format that the key is set on the external data interface. For instance, both mykey
or MYKEY
are converted to Mykey
:
args field |
Is this set? | Description |
---|---|---|
__ce_body |
No | Body |
__ce_headers |
Yes | Header in key-value pair (text-encoded) (key is in canonical format). |
__ce_query |
"" | The query parameter is an empty string. |
Top-level property | No | |
{: caption="args parameter from header parameters when invoking functions"} |
Python example of accessing header data:
import os
def main(args):
try:
header = args["__ce_headers"] # get complete header
# value_1 = args["__ce_headers"]["Key_1"] # get value of the Header parm with "Key_1"
except:
header = "not part of request"
return {
"statusCode": 200,
"body": header,
}
{: screen}
Node.js example of accessing header data:
function main(args) {
// var header_parm_1 = args.__ce_headers.Key_1 //get the value of one header parameter
var header = args.__ce_headers // get all header parameters
return {
statusCode: 200,
"body" : header,
};
}
module.exports.main = main;
{: screen}
{: #fun-args-header-application-json}
A function receives the keys and values of the JSON document as dedicated top-level property parameters.
When no value is set for Content-type
in the HTTP request, then the {{site.data.keyword.codeengineshort}} function uses application/json
as the default.
In addition, the JSON payload is made available to the function as-is (as byte array) in Base64-encoded format in the __ce_body
parameter.
args field |
Is this set? | Description |
---|---|---|
__ce_body |
Yes | Request data (Base64-encoded) |
__ce_headers |
Yes | Content-type is set |
__ce_query |
"" | Empty string |
Top-level property | Yes | Key-value pair for each top-level element (text-encoded) |
{: caption="args parameter from query parameters when invoking functions"} |
Python example of accessing application/json
input data:
import os
def main(args):
try:
body_encoded = args["__ce_body"] # get complete header (base64 encoded)
value_1 = args["key_1"] # get value of the Header parm with "key_1"
except:
value_1 = "not part of request"
return {
"statusCode": 200,
"body": value_1,
}
{: screen}
Node.js example of accessing application/json
input data:
function main(args) {
var body = args.__ce_body // get complete request body (base64 encoded)
var value_1 = args.key_1 // get value of one single key
return {
statusCode: 200,
"body" : value_1,
};
}
module.exports.main = main;
{: screen}
{: #fun-args-request-data-application-octet-stream}
A function receives binary data in Base64-encoded format for the __ce_body
binary data in the __ce_body
parameter:
args field |
Is this set? | Description |
---|---|---|
__ce_body |
Yes | Request data (Base64-encoded) |
__ce_headers |
Yes | Content-type is set |
__ce_query |
"" | Empty string |
Top-level property | No | |
{: caption="args parameter from request data"} |
Python example of accessing application/octet-stream
input data:
import os
import base64
def main(args):
try:
body = base64.b64decode(args['__ce_body']).decode("utf-8") # read binary data into the body variable
except:
body = "not binary data found"
return {
"headers": { "Content-Type": "text/plain" }, # text/plain, if ensured binary data do not conain backslash and double quotes
"statusCode": 200,
"body": body,
}
{: screen}
Node.js example of accessing application/octet-stream
input data:
function main(args) {
var base64EncodedBody = args.__ce_body // get complete request body (base64 encoded)
var body = Buffer.from(args.__ce_body, 'base64').toString('utf-8') // read binary data into the body variable
return {
statusCode: 200,
"body" : body,
};
}
module.exports.main = main;
{: screen}
{: #fun-args-request-data-application-text-plain}
A function receives text type data in Base64-encoded format in the __ce_body
parameter:
args field |
Is this set? | Description |
---|---|---|
__ce_body |
Yes | Request data (text-encoded) |
__ce_headers |
Yes | Content-type is set |
__ce_query |
"" | Empty string |
Top-level property | No | |
{: caption="args parameter from request data"} |
The function's program logic might need to unescape them to re-create the original data.
Python example of accessing text/plain
input data:
import os
def main(args):
body = args['__ce_body'] # get request body, is text encoded (escaped)
return {
"headers": { "Content-Type": "text/plain" },
"statusCode": 200,
"body": body,
}
{: screen}
Node.js example of accessing text/plain
input data:
function main(args) {
var body = args.__ce_body // get complete request body (text encoded)
return {
statusCode: 200,
"body" : body,
};
}
module.exports.main = main;
{: screen}
{: #fun-args-request-data-application-application-x-www-form-urlencoded}
A function receives the complete body data in text-encoded format in the __ce_body
parameter:
args field |
Is this set? | Description |
---|---|---|
__ce_body |
Yes | Request data (text-encoded) |
__ce_headers |
Yes | Content-type is set |
__ce_query |
"" | Empty string |
Top-level property | No | |
{: caption="args parameter from request data"} |
Python example of accessing application/x-www-form-urlencoded
input data:
import os
def main(args):
body = args['__ce_body'] # get request body, is url-encoded (%)
return {
"headers": { "Content-Type": "text/plain" },
"statusCode": 200,
"body": body,
}
{: screen}
Node.js example of accessing application/x-www-form-urlencoded
input data:
function main(args) {
var body = args.__ce_body //get request body, is url-encoded (%)
return {
statusCode: 200,
"body" : body,
};
}
module.exports.main = main;
{: screen}
args
parameter from request data of content-type application/x-www-form-urlencoded
and query parameters
{: #fun-args-request-data-application-application-x-www-form-urlencoded-query}
All combinations of mixed data types are possible, but consider that for HTTP requests with URL query parameters and body parameters, the body parameters take precedence over query parameters.
{: #fun-env-vars}
While function call parameters are derived from the incoming HTTP request, a {{site.data.keyword.codeengineshort}} function also has access to a set of pre-defined environment variables derived from system settings:
- CE_ALLOW_CONCURRENT
- CE_API_BASE_URL
- CE_DOMAIN
- CE_EXECUTION_ENV
- CE_FUNCTION
- CE_PROJECT_ID
- CE_REGION
- CE_SUBDOMAIN
For more information about these environment variables, see Automatically injected environment variables.
Python example of accessing environment variables:
import os
def main(args):
curEnv=dict(os.environ)
return {
"headers": { "Content-Type": "application/json" },
"statusCode": 200,
"body": {
"env": curEnv,
}
}
{: screen}
Node.js example of accessing environment variables:
function main(args) {
var curEnv = process.env; //read the function's env vars
return {
"headers": { "Content-Type": "application/json" },
"statusCode": 200,
"body": {
"env": curEnv,
}
};
}
module.exports.main = main;
{: screen}
{: #fun-internal-response-data-interface}
The internal response data interface depicts how function code provides response data. The {{site.data.keyword.codeengineshort}} functions service processes the data and provides an HTTP response to the caller.
Independent of the programming language, the function code must provide the response data as data structure in the return statement.
Dependent of the programming language, the result object is either a JSON object (Node.js language) or a dictionary (Python language) with the following structure:
Functions result data | Requirement | Description |
---|---|---|
headers |
Optional | A result object in which the keys are header names and the values are strings, numbers, or Boolean values. To send multiple values for a single header, the header's value is an array of the multiple values. No headers are set by default. |
statusCode |
Should (default 200) | A valid HTTP status code. |
body |
Optional (default: empty) | A string that is either plain text, a JSON object or array, or a Base64-encoded string for binary data. The body is considered empty if it is null, an empty string (""), or undefined. |
{: caption="Result object structure in a JSON object (Node.js language) or a dictionary (Python language)"} |
Example result data structure to provide function execution result data:
import os
def main(args):
return {
headers: {
"content_type" : "application/json" ,
"key" , "value"
},
"statusCode": 200,
body: {"myMessage" : "sample message"}
}
{: screen}
Always set body
and statusCode
. The implicit setting of statusCode
and body is only available for compatibility to deprecated {{site.data.keyword.codeengineshort}} functions.
{: note}
{: #fun-headers-element}
The main intention of the headers section is to define the content-type in which the function is going to provide the response data. Result data supports the identical MIME-Type
value as accepted on request.
Depending on the value of the Content-Type
value, the body field in the result structure must contain the following aspects:
Content type | Value that is assigned to the body element | Example |
---|---|---|
application/json |
JSON object | body : { key_1: val1 } |
Not set (default) | String | body : "some text" |
text/* |
String | body : "some text" |
audio/* , example/* , font/' , image/* , video/* , and all remaining types |
Base64-encoded | body: "SGVsbG8gV29ybGQhCg==" |
{: caption="Body field result structure"} |
Also the function code can use key-value pairs in the HTTP response header to provide response data. Therefore, the code must put a unique key name and its value in the headers section of the data structure that is used in the return statement.
Take note of these key name considerations:
- Key names are case-insensitive
- The most recent value assignment of identical key names is used
- Backslashes or blanks are not supported
Take note of these value considerations:
- Values must be text-encoded
- Values must be of data type string or array of string
Example of returning response data as a header key-value pair:
import os
def main(args):
return {
"headers": {
"Content-Type": "text/plain",
"key_1" : "sample_value",
},
"statusCode": 200,
"body": "" ,
}
{: screen}
{: #fun-statusCode-element}
Functions code must explicitly set a status code (default 200
) to inform the caller about the execution result. You can use any valid HTTP return status code (that is, status code 200
to 599
).
The {{site.data.keyword.codeengineshort}} functions service returns the function's status code in the header field (x-faas-actionstatus
) and as an HTTP status code.
If the statusCode
is invalid, then the {{site.data.keyword.codeengineshort}} functions service returns the HTTP return code 422
(Invalid function code cannot be processed) without the x-faas-actionstatus
header field and without extra response data.
If the result size limit for functions is reached, an HTTP status code of 400
is returned to the client.
Example of returning response status code:
import os
def main(args):
return {
"statusCode": 200,
"body": "" ,
}
{: screen}
{: #fun-body-element}
Function code can include the body
section of the return data structure to provide the function's response data. The function code is responsible for delivering the body
element in the return data structure in matching format to the Content-Type
. If the function response Content-Type
is missing, then the function code returns Content-Type: text/plain; charset=utf-8
at the external interface, and the data from the body
field in its unaltered form.
Consider the following body rules, depending on the content-type used.
{: #fun-body-element-application-json}
Function code must provide the value of the body
element as a valid data structure (NodeJs-JSON or Python-Dictionary). The keys and values of the data structure must follow JSON syntax rules so that the {{site.data.keyword.codeengineshort}} service can deliver the response data in the application/json
HTTP response data section.
Backslashes (\
) and double quotation marks ("
) in the data structure are delivered in text-encoded format in the HTTP response data section.
{: note}
Python example of an application/json
response:
import os
def main(args):
# python dictionary
result_body = { "key_1" : "myfolder\myFile" }
return {
"headers": {
"Content-Type": "application/json",
},
"statusCode": 200,
"body": result_body,
}
{: screen}
Node.js example of an application/json
response:
function main(args) {
// JSON data structure
// Note: Backslash must be text encoded
result_body = { "key1" : "myfolder\\myFile"}
return {
statusCode: 200,
headers: {
'Content-Type': "application/json",
},
"body" : result_body ,
};
}
module.exports.main = main;
{: screen}
{: #fun-body-element-application-octet-stream}
The result data for the function code must be Base64-encoded before you add it to the result structure.
Python example of an application/octet-stream
response:
import os
import base64
def main(args):
result_body = "myfolder_myFile"
enc_result_body=base64.encodebytes(result_body.encode()).decode("utf-8").strip()
return {
"headers": {
"Content-Type": "application/octet-stream",
},
"statusCode": 200,
"body": enc_result_body,
}
{: screen}
Node.js example of an application/octet-stream
response:
function main(args) {
var result_body = "###\unreturned body###\n"
var buff = new Buffer(result_body , 'utf8');
return {
statusCode: 200,
headers: {
'Content-Type': "application/octet-stream",
},
"body" : buff.toString('base64') ,
};
}
module.exports.main = main;
{: screen}
{: #fun-body-element-text-plain}
Function code must provide the whole response in a single string. The {{site.data.keyword.codeengineshort}} functions service does not check the response data; the data is transferred to the caller as provided.
Python example of a text/plain
response:
def main(args):
result_body = "myfolder_myFile"
return {
"headers": {
"Content-Type": "text/plain;charset=utf-8",
},
"statusCode": 200,
"body": result_body,
}
{: screen}
{: #fun-body-element-x-www-form-urlencoded}
Function code must help ensure that the response data is URL-encoded before it is added to the result data structure. The {{site.data.keyword.codeengineshort}} functions service does not check the response data for correct syntax. The data is transferred to the caller as provided.
Python example of an application/x-www-form-urlencoded
response:
import os
def main(args):
result_body = "myfolder%20myFile"
return {
"headers": {
"Content-Type": "application/x-www-form-urlencoded",
},
"statusCode": 200,
"body": result_body,
}
{: screen}
{: #fun-external-data-interface-response}
The external data interface definition is derived from standardized HTTP. MIME types in the HTTP Response header are used to define the format of the HTTP response.
{: #fun-external-data-interface-response-mime}
When {{site.data.keyword.codeengineshort}} returns data to the external world, {{site.data.keyword.codeengineshort}} functions can support the same IANA MIME types as described in MIME types.
{: #fun-external-data-interface-response-response-data}
The external data interface returns the response data structure as HTTP header, HTTP status code, and HTTP response data to the caller of the function.
{: #fun-external-data-interface-response-http-header-fields}
The HTTP header contains all key-value pairs that the function code added to the result data structure.
The {{site.data.keyword.codeengineshort}} functions service adds the following key-value pairs:
Field name | Description |
---|---|
x-request-id |
The external request ID for the invoked function. |
x-faas-activation-id |
The {{site.data.keyword.codeengineshort}} functions service internal ID for the invocation. |
x-faas-actionstatus |
The status code the function code set. |
{: caption="Key-value pairs in {{site.data.keyword.codeengineshort}} functions"} |
Header key names are always replied in lowercase. {: note}
{: #fun-external-data-interface-response-http-status-code}
The origin of the HTTP status code can be the {{site.data.keyword.codeengineshort}} functions service itself or the status code set by the function code. The presence of the header field x-faas-statuscode
is the indicator of the origin.
The meaning of the HTTP status code can be read in the {{site.data.keyword.codeengineshort}} functions documentation if the x-faas-actionstatus
header field is not set. However, if the header field is set, then the logic of the function defines the meaning of the HTTP status code.
{{site.data.keyword.codeengineshort}} functions service runs a limited validity check on the response data and content-type, and returns an HTTP status code of 400
if the data format or encoding (generated by the function code) is invalid.
{: note}
{: #fun-external-data-interface-response-http-response-data}
The HTTP response data on the external data interface are identical as supplied in the result data structure by the function code. Only if the MIME-Type
is application/json
then the key and values in the provided JSON response are modified. All backslashes and double quotation marks are escaped.
For example, the caller gets application/json
response data. The function code that provides the response data is as follows:
import os
def main(args):
# python dictionary
result_body = { "key_1" : "myfolder\myFile" }
return {
"headers": {
"Content-Type": "application/json",
"key" : "sample",
},
"statusCode": 200,
"body": result_body,
}
{: screen}
The curl
client that uses the response statusCcode
, the header
fields, and the response data, is as follows:
curl -v -i https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud/
-> Response:
stdout>
* Host default-encoded-response.1lpigeajq5fg.us-east.codeengine.appdomain.cloud:443 was resolved.
* IPv6: 2606:4700:90:0:7554:9304:2cbe:8cbf
* IPv4: 172.65.197.223
* using HTTP/1.x
> GET / HTTP/1.1
> Host: default-encoded-response.1lpigeajq5fg.us-east.codeengine.appdomain.cloud
> User-Agent: curl/8.8.0
> Accept: *//*
* Request completely sent off
< HTTP/1.1 200 OK
< content-type: application/json
< key: sample
< x-faas-actionstatus: 200
< x-faas-activation-id: 5cbab12c-5c6e-4000-96cf-0f7fcb42a979
< x-request-id: e7098271-4780-4893-bbd4-64d4c8d7605e
< content-length: 13
{ "key_1" : "myfolder\\myFile }* Connection
{: screen}
{: #fun-invocation-examples}
Follow these steps to see the creation of a {{site.data.keyword.codeengineshort}} function with the CLI and the usage of the external data interface:
-
Save the following code as
hello.js
locally:function main(args) { return { headers: { content_type: "application/json" }, statusCode: 200, body: {args: args} }; }
{: screen}
-
After you log on to {{site.data.keyword.Bluemix}} and selecting the {{site.data.keyword.codeengineshort}} functions service, create a new function:
ibmcloud ce fn create --name sample --runtime nodejs-18 --inline-code sample.js --cpu 0.5 --memory 2G
{: screen}
Example output:
Creating function 'sample'... OK Run 'ibmcloud ce function get -n sample' to see more details. ibmcloud ce function get -n sample Getting function 'sample'... OK Name: sample ... Status: Ready URL: https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud ...
{: screen}
-
Use the external data interface with a
curl
command to invoke the function:curl -v https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud/
{: screen}
Example output:
{ "args": { "__ce_headers": { "Accept": "*/*", "User-Agent": "curl/7.58.0", "X-Request-Id": "813804ec-ef14-42e9-bce3-c162373defae" }, "__ce_method": "GET", "__ce_path": "/" } }
{: screen}
-
Invoke the function by using query parameters:
curl -v "https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud/?planet1=Mars&planet2=Jupiter"
{: screen}
Example output:
{ "args": { "__ce_headers": { "Accept": "*/*", "User-Agent": "curl/7.58.0", "X-Request-Id": "d03a1af0-bfc8-4a50-be7d-a72040c02cc9" }, "__ce_method": "GET", "__ce_path": "/", "__ce_query": "planet1=Mars&planet2=Jupiter", "planet1": "Mars", "planet2": "Jupiter" } }
{: screen}
The query parameter is available unfolded in the arguments and also unmodified in
__ce_query
. -
Invoke the function by using form data:
curl -H "Content-Type: application/x-www-form-urlencoded" -d 'planet1=Mars&planet2=Jupiter' https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud
{: screen}
Example output:
{ "args": { "__ce_body": "planet1=Mars&planet2=Jupiter", "__ce_headers": { "Accept": "*/*", "Content-Length": "28", "Content-Type": "application/x-www-form-urlencoded", "User-Agent": "curl/7.58.0", "X-Request-Id": "eb3e2179-c396-4eee-98cb-809316f0a765" }, "__ce_method": "POST", "__ce_path": "/" } }
{: screen}
The content of the request body is available unmodified in the
__ce_body
argument with JSON-reserved characters that are being escaped. -
Invoke the function by using a JSON data object:
curl -H "Content-Type: application/json" -d '{"planet1": "Mars", "planet2": "Jupiter"}' https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud
{: screen}
Example output:
{ "args": { "__ce_body": "eyJwbGFuZXQxIjogIk1hcnMiLCAicGxhbmV0MiI6ICJKdXBpdGVyIn0=", "__ce_headers": { "Accept": "*/*", "Content-Length": "41", "Content-Type": "application/json", "User-Agent": "curl/7.58.0", "X-Request-Id": "c06ffcc3-fdae-4430-b881-01d68876c54c" }, "__ce_method": "POST", "__ce_path": "/", "planet1": "Mars", "planet2": "Jupiter" } }
{: screen}
The content of the request body is available unfolded into the function arguments (
args
) with JSON reserved characters that are being escaped and also unmodified as a Base64-encoded string in__ce_body
. -
Invoke the function by using JSON data and query parameters:
curl -H "Content-Type: application/json" -d '{"planet1": "Mars", "planet2": "Jupiter"}' "https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud?planet2=Venus&planet3=Uranus"
{: screen}
Example output:
{ "args": { "__ce_body": "eyJwbGFuZXQxIjogIk1hcnMiLCAicGxhbmV0MiI6ICJKdXBpdGVyIn0=", "__ce_headers": { "Accept": "*/*", "Content-Length": "41", "Content-Type": "application/json", "User-Agent": "curl/7.58.0", "X-Request-Id": "daff83a5-fe53-43ef-8dc4-606e42dd8306" }, "__ce_method": "POST", "__ce_path": "/", "planet1": "Mars", "planet2": "Jupiter", "planet3": "Uranus" } }
{: screen}
The body parameters and query parameters are unfolded into the function arguments (
args
). Body parameters overwrite the query parameters. The request body is available in__ce_body
(Base64-encoded). The query parameters are available in__ce_query
. -
Invoke the function by using text content type:
curl -H "Content-Type: text/plain" -d 'Here we have some text. The JSON special characters like \ or " are escaped.' https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud
{: screen}
Example output:
{ "args": { "__ce_body": "Here we have some text. The JSON special characters like \\ or \ are escaped.", "__ce_headers": { "Accept": "*/*", "Content-Length": "76", "Content-Type": "text/plain", "User-Agent": "curl/7.58.0", "X-Request-Id": "43d259eb-4247-41a9-894a-1dbd98eb16fb" }, "__ce_method": "POST", "__ce_path": "/" } }
{: screen}
The content of the request body is available unmodified in
__ce_body
, but with JSON special characters escaped with\\
or\
. -
Invoke the function by using binary content type:
curl -H "Content-Type: application/octet-stream" -d 'This string is treaded as binary data.' https://sample.1kweru2e873.eu-gb.codeengine.appdomain.cloud
{: screen}
Example output:
{ "args": { "__ce_body": "VGhpcyBzdHJpbmcgaXMgdHJlYWRlZCBhcyBiaW5hcnkgZGF0YS4=", "__ce_headers": { "Accept": "*/*", "Content-Length": "38", "Content-Type": "application/octet-stream", "User-Agent": "curl/7.58.0", "X-Request-Id": "a90826e0-db13-4b8a-809f-60cea2a27d96" }, "__ce_method": "POST", "__ce_path": "/" } }
{: screen}