Native Language Workflows
Since the framework currently supports Node.js and Bun.js ecosystems, the native languages currently supported are TypeScript and JavaScript. This allows users to create custom functions. A native language workflow enables us to incorporate additional features using JavaScript or TypeScript, where we have the capability to implement intricate business logic.
Example Typescript function
import { GSCloudEvent, GSContext, PlainObject } from "@godspeedsystems/core";
import Pino from 'pino';
export default function (ctx: GSContext, args: PlainObject) {
const {
inputs: {
data: {
params, body, query, user, headers
}
},
childLogger,
logger,
outputs,
functions,
datasources,
mappings
}: {
inputs: GSCloudEvent,
childLogger: Pino.Logger, // Define CustomLogger if necessary
logger: Pino.Logger,
outputs: PlainObject, // Adjust the type accordingly
functions: PlainObject, // Adjust the type accordingly
datasources: PlainObject, // Adjust the type accordingly
mappings: PlainObject
} = ctx;
// Will print with workflow_name and task_id attributes
childLogger.info('Server is running healthy');
// Will print without workflow_name and task_id attributes
logger.info('Arguments passed %o', args);
logger.info('Inputs object \n user %o query %o body %o headers %o params %o', user, query, body, headers, params);
logger.info('Outputs object has outputs from previous tasks with given ids %o', Object.keys(outputs));
logger.info('Datasources object has following datasource clients %o', Object.keys(datasources));
logger.info('Total functions found in the project %s', Object.keys(functions).length);
// Returning only data
return 'Its working! ' + body.name;
//SAME AS
return {
data: 'Its working! ' + body.name,
code: 200,
// success: true,
// headers: undefined
}
//SAME AS
return {
data: 'Its working! ' + body.name,
code: 200,
success: true,
// headers: undefined
}
//SAME AS
return {
data: 'Its working! ' + body.name,
code: 200,
success: true,
headers: undefined
}
// SAME AS
return new GSStatus(true, 200, undefined, 'Its working! ' + body.name, undefined);
}
For seeing how framework handles data returned from a function, including calculation of code
, success
and data
, read this section at the bottom of the page.
GSContext
The above is a sample of how JS/TS functions are authored. Functions which are called as event handlers or functions which are called by other YAML workflows have two arguments passed to them
- GSContext - carries the loaded components of this project, and as well the inputs of the current event.
- Arguments - a key-value object (JSON)
args
The second parameter of the function call is args. This parameter is useful when this function is called from a YAML workflow in Godspeed. The args
passed in the yaml task of the caller YAML workflow is passed as args
here. It is a JSON object.
Caller YAML function
summary: some workflow
tasks:
- id: first_task
fn: some_function
args:
name: mastersilv3r
Callee Typescript function
export default function (ctx: GSContext, args: PlainObject) {
ctx.logger.info(args.name); //Prints 'mastersilv3r'
}
More about GSContext
Every function/workflow has access to the ctx object, which is passed as an argument, and furthermore, you can access its properties by destructuring it.
Check the code of GSContext interface here. GSContext has the contextual information of your current workflow and is available to the event handlers (functions
). It is passed to any sub workflows subsequently called by the event handler.
It includes all the context specific information like tracing information, actor, environment, headers, payload etc.
Every information you need to know or store about the event and the workflow executed so far, and as well the loaded functions
, datasources
, logger
, childLogger
, config
, mappings
etc, is available in the GSContext
object.
// Everything you need within a workflow, whether in native languages like JS/TS, or in yaml workflows and tasks.
export class GSContext { //span executions
outputs: { [key: string]: GSStatus; }; //DAG result. This context has a trace history and responses of all instructions in the DAG, which are are stored in this object against task ids
log_events: GSLogEvent[] = [];
config: PlainObject; //The config folder with env vars, default, and other config files. We use node-config module for Nodejs for the same.
datasources: PlainObject; //All the datasource exported clients
log_events: GSLogEvent[] = []; //All the errors during an event handler workflow execution are captured in this list. Framework does not do anything with this. But a developer may want to have access to the errors that happened.
config: PlainObject; //app config
datasources: PlainObject; //app config
mappings: PlainObject; // The static mappings of your project under /mappings
functions: PlainObject; //All the functions you have written in /functions + all the Godspeed's YAML DSL functions
//like com.gs.each_parallel
plugins: PlainObject; // The utility functions to be used in scripts. Not be confused with eventsource or datasource as plugin.
exitWithStatus?: GSStatus; // Useful when a YAML workflow is being executed. If this is set to non null value containing a GSStatus, the workflow will exit with this status. This will apply to only the immediate yaml workflow. But not its caller workflow.
logger: pino.Logger; // For logging using pino for Nodejs. This has multiple useful features incudign biding some key values with the logs that are produced.
childLogger: pino.Logger; //Child logger of logger with additional binding to print {workflow_name, task_id} with every log entry
forAuth?: boolean = false; //Whether this native or yaml workflow is being run as parth of the authz tasks
}
Inputs
Inputs Provide you all the Information you passed to event like headers
, params
, query
, params
(path params), body
& user
(parsed JWT information).
const {inputs} = ctx;
const body = inputs.data.body;
Outputs
To access outputs of tasks executed before the current task, developer can destruct ctx object just like how inputs and datasources.If we have more then one task, we can access first task outputs in second task with Outputs object. we should access first task output by useing it's id.
const {outputs} = ctx;
const firstTaskOutput = outputs[firstTaskId]
Accessing Datasource Clients
With datasources we can access all datasources, their clients and methods.
const { datasources} = ctx;
const responseData = await datasources.mongo.client.Restaurant.create({
data: inputs.body
})
ChildLogger
With childLogger you have accessibility to Pino logger instance with context information set - for example the log.attributes
set in eventsource or event level.
const { childLogger} = ctx;
childLogger.info('inputs: %o', inputs.body);
Returning from a function
GSStatus
Developers can return pure JSON object in response, or return GSStatus if they use Typescript. The GSStatus is a built-in class in Godspeed. We invoke it when we're prepared to define an API response manually and dispatch it. GSStatus has the below properties.
success: boolean;
code?: number;
message?: string;
data?: any;
headers?: {
[key: string]: any;
};
We return with GSStatus as below
return new GSStatus(true, 200, 'OK', responseData, headers);
Different ways to return from a event handler
// Returning only data
return 'Its working! ' + body.name;
//SAME AS
return {
data: 'Its working! ' + body.name,
code: 200,
message: 'OK', //HTTP protocol message to be returned from service
// success: true,
// headers: undefined
}
//SAME AS
return {
data: 'Its working! ' + body.name,
code: 200,
message: 'OK', //HTTP protocol message to be returned from service
success: true,
// headers: undefined
}
//SAME AS
return {
data: 'Its working! ' + body.name,
code: 200,
message: 'OK', //HTTP protocol message to be returned from service
success: true,
headers: undefined
}
// SAME AS returning GSStatus like this
return new GSStatus(true, 200, 'OK', 'Its working! ' + body.name, headers);
Check event handler response to know how framework handles GSStatus.
Invoking functions from JS/TS functions
All functions within a Godspeed project, including those written in YAML, JavaScript (JS), or TypeScript (TS), are accessible through the
ctx.functions
object.Ofcourse you can also
import
them in the standard Typescript or Javascript waySimilarly, all datasource clients initialised in a Godspeed project are conveniently available under the
ctx.datasources
object.export default async function (ctx: GSContext, args: any) {
//Calling functions (yaml, js, ts) from within a ts/js function, in a meta framework's project's functions folder, all project functions are available under ctx.functions.
const res = await ctx.functions['com.gs.helloworld2'](ctx, args);
//Same As
const res = await require('com/gs/helloworld2')(ctx, args);
// Calling datasource functions. All datasources are available under ctx.datasources hood.
// OPTION 1
// Every datasource exposes a client key. The client may be a single instance like in case of Axios, or multiple datasource client instances like in case of AWS, or database models like in case of Mongoose (depending on the plugin used).
const res = await ctx.datasources.aws.client.s3.listBuckets(args);
// OPTION 2
// All datasources have an execute method. This may be preferable in case you want to utlise the full capabilities of the plugin wrapped over the native clients, like error handling checks and response codes, retries, caching etc.
const res = await ctx.datasources.aws.execute(ctx, {
//Pass exactly same args as this aws service's method takes
...args,
meta: {entityType: 's3', method: 'listBuckets'}
//Along with args, pass meta object
// meta can contain {entityType, method}
});
if (!res.success) {
return new GSStatus(false, res.code || 500, undefined, {message: "Internal Server Error", info: res.message})
}
//If a developer only returns data without setting keys like "success" or "code" in the response,
// the framework assumes it is just the data.
//In such cases, the response code defaults to 200, and success is assumed to be true.
return res
// works same as return new GSStatus(true, 200, undefined, res );
}
Handling event handler return
By default, all the framework defined functions or developer written functions, have to return either GSStatus or data.
Now lets see how the framework qualifies your return as GSStatus or simple data.
The framework sees that your returned data has one of code
or success
meta-keys.
Non Authz (normal) workflows
i. If both are present, it looks for the other GSStatus keys and set them.
ii. If only code is present and lies between 200-399, then success is assumed to be true else false. It looks for the other GSStatus keys and set them.
iii. If only success is present, then code is assumed to be 200. It looks for the other GSStatus keys and set them.
iv. If it doesn't find any of these keys, it assumes all that you have returned is intended to be GSStatus.data then it adds code: 200
and success: true
internally to your response and create a GSStatus
out of it to pass on to next tasks or workflows.
Authz workflows
Check reponse handling in case of authorization workflows.
You can study the code here to understand both of the above scenarios better.