From 4ec60751e865b31bdab259600612e78af639dd5c Mon Sep 17 00:00:00 2001 From: Khuda Dad Nomani <32505158+KhudaDad414@users.noreply.github.com> Date: Fri, 26 Jan 2024 13:46:17 +0000 Subject: [PATCH] docs: improve docs (#699) --- docs/pages/authentication.md | 97 +++++------ docs/pages/bearerToken.md | 43 +++-- docs/pages/env-vars-config.md | 217 ++++++++---------------- docs/pages/function-lifecycle-events.md | 96 +++++------ docs/pages/glee-auth-intro.md | 9 +- docs/pages/glee-template.md | 98 ----------- docs/pages/httpApiKey.md | 36 ++-- docs/pages/index.md | 130 +------------- docs/pages/installation.md | 79 +++++---- docs/pages/your-first-glee-app.md | 125 ++++++++++++++ 10 files changed, 394 insertions(+), 536 deletions(-) delete mode 100644 docs/pages/glee-template.md create mode 100644 docs/pages/your-first-glee-app.md diff --git a/docs/pages/authentication.md b/docs/pages/authentication.md index 841bbfca7..031810402 100644 --- a/docs/pages/authentication.md +++ b/docs/pages/authentication.md @@ -1,45 +1,44 @@ --- -title: 'Authentication functions' +title: 'Authentication Functions' weight: 70 --- -# Getting started with Authentication functions +# Getting Started with Authentication Functions -Authentication in Glee can be done using authentication functions. Authentication functions are files that export either one or both of the `clientAuth` and `serverAuth` Node.js functions: +Authentication in Glee can be implemented using authentication functions. These functions are files that export one or both of the following Node.js functions: `clientAuth` and `serverAuth`: ```js /* websocket.js */ export async function serverAuth({ authProps, done }) { - //server auth logic + // Server authentication logic } export async function clientAuth({ parsedAsyncAPI, serverName }) { - //client auth logic + // Client authentication logic } ``` -Glee looks for authentication files in the `auth` directory by default but it can be configured using [glee config file](env-vars-config). -The name of the authentication file should be the name of the targeted server that the authentication logic should work for. +Glee searches for authentication files in the `auth` directory by default. However, this can be configured using the [glee config file](env-vars-config). The authentication file's name should match the targeted server for which the authentication logic is intended. -## Supported Authentication Values in asyncapi.yaml file +## Supported Authentication Values in the asyncapi.yaml File -AsyncAPI currently supports a variety of authentication formats as specified in the [documentation](https://www.asyncapi.com/docs/reference/specification/v3.0.0#securitySchemeObject), however Glee supports the following authentication schemas. +AsyncAPI supports a variety of authentication formats as specified in its [documentation](https://www.asyncapi.com/docs/reference/specification/v3.0.0#securitySchemeObject). Glee, however, supports the following authentication schemas: - userPassword - http ("bearer") - httpApiKey - Oauth2 -A sample `asyncapi.yaml` for a **server** with security requirements and a `userPassword` security schemes is shown below: +Below is an example of a `asyncapi.yaml` file for a **server** with security requirements and a `userPassword` security scheme: ```yaml -##server asyncAPI schema +## Server AsyncAPI Schema asyncapi: 3.0.0 info: - title: AsyncAPI IMDB server + title: AsyncAPI IMDB Server version: 1.0.0 - description: This app is a dummy server that would stream the trending/upcoming anime. + description: This app is a dummy server that streams trending/upcoming anime. servers: trendingAnimeServer: host: 'localhost:8081' @@ -53,13 +52,12 @@ components: securitySchemes: userPass: type: userPassword - ``` -A sample `asyncapi.yaml` for a **client** that implements some of the requirements of the server above is as follows: +Here's an example for a **client** that implements some requirements of the server mentioned above: ```yaml -##client asyncAPI schema +## Client AsyncAPI Schema servers: trendingAnime: host: localhost:8081 @@ -78,29 +76,28 @@ components: securitySchemes: userPass: type: userPassword - ``` -Glee can act as both a server and a client. So the need for `serverAuth` and `clientAuth`. Glee acts as a client when the server name is included in the `x-remoteServers` property in the `asyncapi.yaml` file. +Glee can function as both a server and a client. Hence, the need for both `serverAuth` and `clientAuth` functions arises. Glee acts as a client when the server name is included in the `x-remoteServers` property in the `asyncapi.yaml` file. -When Glee acts as a client, it can connect to a Glee server, and when Glee acts as a server it accepts connections from other Glee clients. Hence a Glee application can both accept connections from clients while also sending requests to other Glee applications (servers) at the same time. +When Glee operates as a client, it can connect to a Glee server. Conversely, as a server, it accepts connections from other Glee clients. Thus, a Glee application can accept connections from clients while also sending requests to other Glee servers. -When a security requirement is specified in the `asyncapi.yaml` file and Glee acts as a server, the `serverAuth` function should be implemented, if Glee acts as a client then the `clientAuth` function should be implemented. If Glee is being used as both client and server, then it should have both the `clientAuth` and `serverAuth` functions. +If a security requirement is specified in the `asyncapi.yaml` file, and Glee acts as a server, the `serverAuth` function should be implemented. If Glee acts as a client, then `clientAuth` should be implemented. If Glee is used as both client and server, both functions are necessary. ## Server Authentication in Glee -The `serverAuth` function takes an argument that can be destructured as follows +The `serverAuth` function takes an argument that can be destructured as follows: | Attribute | Description | | ---------- | --------------------------------------------------------------- | -| done | The done function that tells the server to proceed. | -| authProps | The authentication parameters recieved from the client. | -| serverName | The name of the server/broker from which the event was emitted. | -| doc | The parsedAsyncAPI schema | +| done | The function that signals the server to proceed. | +| authProps | The authentication parameters received from the client. | +| serverName | The name of the server/broker emitting the event. | +| doc | The parsed AsyncAPI schema. | -#### done() function +#### done() Function -The `done()` parameter in the `serverAuth` function allows the broker/server to know what to do next depending on the boolean value you pass to it. +The `done()` parameter in the `serverAuth` function signals to the broker/server what action to take next, based on the boolean value passed. ```js /* websocket.js */ @@ -113,22 +110,22 @@ export async function serverAuth({ authProps, done }) { } } ``` + **Parameters for done():** -*Authentication Result (Boolean): true for success, false for failure.* +- Authentication Result (Boolean): `true` for success, `false` for failure. -When `true` is passed to the done parameter, the server/broker knows to go ahead and allow the client to connect, which means authentication has succeeded. However if the `done` parameter is called with `false` then the server knows to throw an error message and reject the client, which means authentication has failed. +Passing `true` to the `done` parameter indicates that authentication has succeeded, and the server/broker can proceed to allow the client to connect. Conversely, if `false` is passed, the server will reject the client, indicating failed authentication. -`done()` should always be the last thing called in a `serverAuth` function, Glee won't execute any logic beyond the `done()` call. +The `done()` call should always be the last in the `serverAuth` function, as Glee will not execute any logic beyond this call. #### authProps -`authProps` implements a couple of methods that allows the server to retrieve the authentication parameters from the client, below are the current available methods; +The `authProps` parameter includes methods for the server to retrieve authentication parameters from the client. The current available methods are as follows: ```js export async function serverAuth({ authProps, done }) { - //some network request - + // Some network request authProps.getOauthToken() authProps.getHttpAPIKeys('api_key') authProps.getToken() @@ -140,23 +137,23 @@ export async function serverAuth({ authProps, done }) { | Method | Description | | ---------------------- | ------------------------------------------------------------------------------------------------ | -| `getOauthToken()` | returns the oauth authentication parameter | -| `getHttpAPIKeys(name)` | returns the HttpAPIKeys parameter with the specified name from either headers or query parameter | -| `getToken()` | returns the http bearer token parameter | -| `getUserPass()` | returns username and password parameters | +| `getOauthToken()` | Returns the OAuth authentication parameter. | +| `getHttpAPIKeys(name)` | Returns the HttpAPIKeys parameter with the specified name from either headers or query parameter | +| `getToken()` | Returns the HTTP bearer token parameter. | +| `getUserPass()` | Returns username and password parameters. | ## Client Authentication in Glee -The `clientAuth` function also takes an argument, and it's argument can be destructured as follows +The `clientAuth` function also takes an argument that can be destructured as follows: | Attribute | Description | | -------------- | ------------------------------------------------------------------------------------- | -| parsedAsyncAPI | The parsedAsyncAPI schema. | -| serverName | The name of the server/broker from with the authentication parameters are being sent. | +| parsedAsyncAPI | The parsed AsyncAPI schema. | +| serverName | The server/broker's name from which the authentication parameters are being sent. | -### Possible authentication parameters +### Possible Authentication Parameters -The possible authentication parameters are shown in the code snippet below: +The code snippet below illustrates the possible authentication parameters: ```js export async function clientAuth({ serverName }) { @@ -165,18 +162,18 @@ export async function clientAuth({ serverName }) { oauth: process.env.OAUTH2, apiKey: process.env.APIKEY, userPass: { - user: process.env.user, - password: process.env.password, + user: process.env.USER, + password: process.env.PASSWORD, }, } } ``` -The name of the authentication parameters should be the same as **the names specified in the `asyncapi.yaml` file.** +The names of the authentication parameters should match **the names specified in the `asyncapi.yaml` file**. -| auth type | values | +| Auth Type | Values | | ------------------------------------- | ---------------------------------------------------------------------- | -| http bearer (JWT) | Value should be a JWT string | -| Oauth2 | The value should should be a string | -| httpApiKey in headers or query params | The value should be a string | -| userPass | The value should be an object with the user and password as properties | +| HTTP bearer (JWT) | Value should be a JWT string. | +| OAuth2 | Value should be a string. | +| httpApiKey in headers or query params | Value should be a string. | +| userPass | Value should be an object with the user and password as properties. | \ No newline at end of file diff --git a/docs/pages/bearerToken.md b/docs/pages/bearerToken.md index db22c2b6c..902c5a7fb 100644 --- a/docs/pages/bearerToken.md +++ b/docs/pages/bearerToken.md @@ -1,21 +1,21 @@ --- -title: 'Http Authentication(Bearer Token)' +title: 'HTTP Authentication (Bearer Token)' weight: 80 --- -## Getting started with Bearer Token authentication +## Getting Started with Bearer Token Authentication -Bearer Token authentication is one of the most popular forms of authentication and is widely used because of its perceived security. This guide will walk through how to implement bearer token authentication in Glee. +Bearer Token authentication is one of the most popular forms of authentication and is widely used due to its perceived security. This guide will walk you through how to implement bearer token authentication in Glee. -A sample `asyncapi.yaml` for a server with security requirements and user password security scheme is shown below: +Below is a sample `asyncapi.yaml` for a server with security requirements and a user password security scheme: ```yaml -##server asyncAPI schema +## Server AsyncAPI Schema asyncapi: 3.0.0 info: - title: AsyncAPI IMDB server + title: AsyncAPI IMDB Server version: 1.0.0 - description: This app is a dummy server that would stream the trending/upcoming anime. + description: This app is a dummy server that streams trending/upcoming anime. servers: trendingAnimeServer: host: 'localhost:8081' @@ -37,7 +37,7 @@ components: A sample `asyncapi.yaml` for a client that implements some of the requirements of the server above: ```yaml -##client asyncAPI schema +## Client AsyncAPI Schema servers: trendingAnime: host: localhost:8081 @@ -61,49 +61,46 @@ components: ``` -The Client asyncapi.yaml file **does't need to implement all the security requirements in the server, it only needs to implement the ones that it uses like *http (bearer token)* here.** +The Client `asyncapi.yaml` file **doesn't need to implement all the security requirements of the server; it only needs to implement the ones it uses, like *http (bearer token)* here.** ### Client Side -Following the client `asyncapi.yaml` file above, create a file named `trendingAnime.ts` in the `auth` directory, since that is the server that has the security Property. +Following the client `asyncapi.yaml` file above, create a file named `trendingAnime.ts` in the `auth` directory, since that is the server that has the security property. ```bash touch auth/trendingAnime.ts ``` -When using the `bearer` security scheme, it is important that you pass the parameters as follows: +When using the `bearer` security scheme, pass the parameters as follows: ```js -export async clientAuth({ parsedAsyncAPI, serverName }) { +export async function clientAuth({ parsedAsyncAPI, serverName }) { return { token: process.env.TOKEN } } ``` -Glee will utilize the `token` for server authentication, employing it in the header with the format: Authorization: Bearer {token}. +Glee will utilize the `token` for server authentication, employing it in the header with the format: `Authorization: Bearer {token}`. -### Server side +### Server Side -From the server `asyncapi.yaml` file above, create a file named `trendingAnimeServer.ts` in the `auth` directory, since that is the server that has the security Property. +From the server `asyncapi.yaml` file above, create a file named `trendingAnimeServer.ts` in the `auth` directory, since that is the server that has the security property. ```bash touch auth/trendingAnimeServer.ts ``` -On the server side, you can retrieve the values as follows +On the server side, you can retrieve the values as follows: ```js -export async serverAuth({ authProps, done }) { +export async function serverAuth({ authProps, done }) { authProps.getToken() - // your authentication logic here... - done(true|false) + // Your authentication logic here... + done(true || false) } ``` -So, `getToken()` returns a string which contains the token that is sent from the client. - - - +So, `getToken()` returns a string containing the token sent from the client. \ No newline at end of file diff --git a/docs/pages/env-vars-config.md b/docs/pages/env-vars-config.md index 0dddf1612..9c8590e28 100644 --- a/docs/pages/env-vars-config.md +++ b/docs/pages/env-vars-config.md @@ -1,205 +1,136 @@ --- -title: Environment variables and Configuration file +title: Configuring Environment Variables and Configuration File weight: 50 --- -# Environment Variables +## Environment Variables -Glee provides a few environment variables for you to customize the Glee application's behavior according to your specific requirements: +Glee provides several environment variables that allow you to tailor your application's behavior to fit specific needs: -|Variable|Description|Example| -|---|---|---| -|GLEE_SERVER_NAMES|A comma-separated list of the servers to load on startup.|`GLEE_SERVER_NAMES=websockets,mosquitto`| -|GLEE_SERVER_CERTS|A comma-separated list of `${serverName}:${pathToCertificateFile}`. These are the certificates to use when establishing the connection to the given server.|`GLEE_SERVER_CERTS=mosquitto:mosquitto.org.crt`| -|GLEE_SERVER_VARIABLES|A comma-separated list of `${serverName}:${serverVariable}:${value}`. These are the values to use for each server variable.|`GLEE_SERVER_VARIABLES=websockets:namespace:public`| +|Variable|Purpose|Example Usage| +|--------|-------|-------------| +|GLEE_SERVER_NAMES|Specifies a list of servers to initialize at startup, separated by commas.|`GLEE_SERVER_NAMES=websockets,mosquitto`| +|GLEE_SERVER_CERTS|Indicates server-specific certificate files in a `${serverName}:${pathToCertificateFile}` format, separated by commas.|`GLEE_SERVER_CERTS=mosquitto:mosquitto.org.crt`| +|GLEE_SERVER_VARIABLES|Sets server variables in a `${serverName}:${serverVariable}:${value}` format, separated by commas.|`GLEE_SERVER_VARIABLES=websockets:namespace:public`| -### Support for multiple .env files -Glee has support for loading variables from `.env.local` into `process.env`. -This is useful for storing secret environment variables needed in development while keeping them out of the repository. -However, sometimes you might want to add some defaults for the `development` or `production` environment. You can do that by creating files with the following names: -`.env.development` or `.env.production` +### Handling Multiple .env Files +Glee supports loading variables from `.env.local` directly into `process.env`. This feature is handy for keeping secrets out of your repository during development. You can also set environment-specific defaults in `.env.development` or `.env.production`. -`.env.local` always overrides any other existing `.env*` file. +`.env.local` takes precedence over other `.env*` files. -You can change the environment by setting the `NODE_ENV` variable to `development` or `production`. +Switch between `development` and `production` environments by setting the `NODE_ENV` variable accordingly. -## Configuring Glee +## Customizing Glee Settings -Glee comes with sensible defaults so you don't have to worry about configuration. However, sometimes you may want to change the behavior or customize Glee in different ways. For that purpose, you can use the `glee.config.js` file. +While Glee comes with defaults for ease of use, you may want to customize settings for specific needs. This is where `glee.config.js` comes into play. -### Configuration file +### The Glee Config File -Glee's config file is a JavaScript file that exports an async function. Something like this: +`glee.config.js` is a JavaScript file exporting an asynchronous function, structured as follows: ```js export default async function () { - // More stuff here... + // Configuration details go here... } ``` -This function must return an object with the following shape: +This function should return an object with configurable properties: ```js export default async function () { return { glee: {}, - kafka: {}, websocket: {}, - mqtt: {}, cluster: {}, http: {} } } - ``` -Here is an example of a `glee.config.js` file for reference: +For example, a typical `glee.config.js` might look like this: ```js export default async function () { return { - glee: { // Glee core configurations + glee: { // Core Glee configurations lifecycleDir: './lifecycle', functionsDir: './functions', asyncapiFilePath: './asyncapi.json', - logs: { // you can change the defualt behaviour of glee which logs everything by default. - incoming: 'channel-only', // only logs the channel not message payload. - outgoing: 'none', //log nothing. + logs: { // Adjust default logging behavior + incoming: 'channel-only', // Logs only the channel, not the message payload + outgoing: 'none', // Disables outgoing logs } }, docs: { - enabled: true, // Enable/Disable documentation generation - folder: 'docs', // Folder where you want the output of your docs to reside. - template: '@asyncapi/markdown-template' // Type of template you want to use. - } + enabled: true, // Toggles documentation generation + folder: 'docs', // Destination folder for docs + template: '@asyncapi/markdown-template' // Specifies the documentation template + }, ws: { server: { - httpServer: customServer, // A custom HTTP server of your own. - adapter: "native", // Default. Can also be 'socket.io' or a reference to a custom adapter. - port: process.env.PORT, + httpServer: customServer, // Custom HTTP server + adapter: "native", // Defaults to 'native', can be 'socket.io' or a custom adapter + port: process.env.PORT, // Server port } }, cluster: { - adapter: "redis", - name: "cluster", // Default. Name of your cluster. - url: "redis://localhost:6379", // Server URL used by adapter for clustering - }, - mqtt: { - auth: ({serverName, parsedAsyncAPI}) => { - if (serverName === 'mqtt') { - return { - cert: async () => fs.readFileSync('./cert') - clientId: '123', - username: 'user1' - password: 'pass12' - } - } - } + adapter: "redis", // Cluster adapter, default is Redis + name: "cluster", // Cluster name + url: "redis://localhost:6379", // URL for the cluster server (Redis in this case) }, http: { server: { - httpServer: customServer, // A custom HTTP server of your own. + httpServer: customServer, // Custom HTTP server adapter: 'native', - port: process.env.PORT, + port: process.env.PORT, // Server port }, - client: { - auth: { - token: process.env.TOKEN - }, - query: { - foo: 'bar' - }, - body: { - foo: 'bar' - } - } } }; } ``` -Inside the return statement, you can specify the following options: -#### Glee Core Configurations -These configurations apply to Glee itself, rather than any specific protocol. +In the return statement, configure the following options: + +#### Core Glee Configurations +These settings are specific to Glee itself. -|Field|Default|Description| +|Field|Default|Purpose| |--|--|--| -|glee.gleeDir|`.glee`|Sets the Glee directory. Your sources will be compiled here.| -|glee.lifecycleDir|`lifecycle`|Path to the directory that stores your [lifecycle events](./lifecycle-events.md).| -|glee.functionsDir|`functions`| Path to the directory that stores your [functions](./functions.md).| -|glee.asyncapiFilePath|`asyncapi.(yaml \| yml \| json)`| Path to your AsyncAPI file.| -|glee.logs| default | glee logs channel and payload by default. you can change this behaviour for incoming and outgoing messages.| -|glee.logs.incoming| "all" | Supported values are `channel-only` and `none`.| -|glee.logs.outgoing| "all" | Supported values are `channel-only` and `none`.| - -#### Generating Documentation -|Field|Description| -|--|--| -|docs.enabled|This flag enables/disables the docs generation functionality. -|docs.folder|The dedicated folder you want your generated docs to reside. -|docs.template|The AsyncAPI template you wanna use for generating your documentation. -#### Websocket Server -|Field|Description| +|glee.gleeDir|`.glee`|Determines the Glee directory for compiled sources.| +|glee.lifecycleDir|`lifecycle`|Specifies the path to [lifecycle events](./lifecycle-events.md).| +|glee.functionsDir|`functions`|Designates the path to [functions](./functions.md).| +|glee.asyncapiFilePath|`asyncapi.(yaml \| yml \| json)`|Path to your AsyncAPI file.| +|glee.logs|default|Configures logging for incoming and outgoing messages.| +|glee.logs.incoming|"all"|Options: `channel-only`, `none`.| +|glee.logs.outgoing|"all"|Options: `channel-only`, `none`.| + +#### Documentation Configuration +|Field|Purpose| |--|--| -|ws.server|Websocket server-specific configurations| -|ws.server.adapter| The Glee adapter to use for the WebSocket server. Defaults to a "native" WebSocket implementation. Other allowed values are `socket.io` (to use the [Socket.IO](https://socket.io/) Glee adapter) or a reference to a custom adapter.| -|ws.server.httpServer| A custom HTTP server of your own. E.g., an [Express](https://expressjs.com/en/4x/api.html) server or any object that implements the [http.Server](https://nodejs.org/api/http.html#http_class_http_server) interface. | -|ws.server.port| The port to use when binding the WebSocket server. This is useful when your server is behind a proxy and the port exposed for consumption is not the same as the port your application should be bound to. Defaults to the port specified in the selected AsyncAPI server.| -#### Cluster -|Field|Description| +|docs.enabled|Enables or disables documentation generation.| +|docs.folder|Specifies the output directory for documentation.| +|docs.template|Determines the AsyncAPI template for docs generation.| + +#### WebSocket Server Configuration +|Field|Purpose| |--|--| -|cluster.adapter| The Glee cluster adapter to use for communication between instances. Defaults to Redis Pub/Sub ("redis"). Can be a reference to a custom adapter.| -|cluster.name|The name of the cluster. Defaults to "cluster".| -|cluster.url|The url of the server to be used by the adapter. In case of "redis" adapter, it's the url of the Redis server.| -#### MQTT -|Field|Description| -|---|---| -|mqtt.auth| MQTT authentication configuration| -|mqtt.auth.cert| Client certificate -|mqtt.auth.clientId| MQTT client Id for authentication -|mqtt.auth.username| username parameter -|mqtt.auth.password| password parameter -#### Kafka -|Field|Description| -|---|---| -|kafka.auth| Kafka authentication configuration| -|kafka.auth.key | Kafka Broker Key -|kafka.auth.cert| Client certificate -|kafka.auth.clientId| Kafka client Id for authentication -|kafka.auth.rejectUnauthorized | Boolean flag for accepting the valid SSL certificates -|kafka.auth.username| The username to use during authentication. -|kafka.auth.password| The password to use during authentication. -#### HTTP Server -|Field|Description| +|ws.server|WebSocket server-specific settings.| +|ws.server.adapter|Selects the WebSocket server adapter: `native`, `socket.io`, or a custom one.| +|ws.server.httpServer|A custom HTTP server instance.| +|ws.server.port + +|The port for the WebSocket server.| + +#### Cluster Configuration +|Field|Purpose| |--|--| -|http.server|HTTP server-specific configurations| -|http.client|HTTP client-specific configurations| -|http.server.adapter| The Glee adapter to use for the HTTP server. Defaults to a "native" HTTP implementation.| -|websocket.server.port| The port to use when binding the HTTP server. This is useful when your server is behind a proxy and the port exposed for consumption is not the same as the port your application should be bound to. Defaults to the port specified in the selected AsyncAPI server.| -|http.client.auth| Authentication/Authorization configuration for the client| -|http.client.auth.token| HTTP Authentication header| -|http.client.query| Query object for the client to send| -|http.client.body| Body object for the client to send -#### Auth Config -Most clients like `ws`,`kafka`, and `mqtt` have auth fields that are used for passing auth parameters. All these configurations can be an object or a function that returns the specific object defined by each protocol. +|cluster.adapter|Chooses the cluster communication adapter (default: Redis Pub/Sub).| +|cluster.name|The cluster's name.| +|cluster.url|URL of the server used by the cluster adapter.| -```js -export default async function() { - ws: { - client: { - auth: { - token: process.env.TOKEN - } - } - }, - mqtt: { - auth: ({serverName, parsedAsyncAPI}) => { - if (serverName === 'mqtt') { - return { - cert: fs.readFileSync('./cert', 'utf-8') - } - } - } - } -} -``` +#### HTTP Server Configuration +|Field|Purpose| +|--|--| +|http.server|HTTP server-specific settings.| +|http.server.adapter|Selects the HTTP server adapter.| +|http.server.port|The port for the HTTP server.| diff --git a/docs/pages/function-lifecycle-events.md b/docs/pages/function-lifecycle-events.md index 7e4a1bd25..3d10988d0 100644 --- a/docs/pages/function-lifecycle-events.md +++ b/docs/pages/function-lifecycle-events.md @@ -1,5 +1,5 @@ --- -title: Function and Lifecycle events +title: Functions and Lifecycle Events weight: 40 --- @@ -13,42 +13,43 @@ export default async function (event) { } ``` -Functions take a single argument, which is the event received from a broker or a client, depending which kind of API you're building. The `event` argument has the following shape: +Functions take a single argument, which is the event received from a broker or a client, depending on the type of API you're building. The `event` argument has the following structure: |Attribute|Description| -|----|----| -|payload|The payload/body of the received event. -|headers|The headers/metadata of the received event. -|channel|The name of the channel/topic from which the event was read. -|serverName|The name of the server/broker from which the event was received. +|---------|-----------| +|payload|The payload/body of the received event.| +|headers|The headers/metadata of the received event.| +|channel|The name of the channel/topic from which the event was read.| +|serverName|The name of the server/broker from which the event was received.| -Functions may return an object to tell Glee what to do next. For instance, the following example sends a greeting message to `development` server: +Functions may return an object to instruct Glee on what action to take next. For instance, the following example sends a greeting message to the `development` server: ```js /* onHello.js */ export default async function (event) { return { send: [{ - server: 'developement', + server: 'development', channel: 'greets', payload: 'Greetings! How is your day going?' }] - } + }; } ``` |Attribute|Type|Description| -|---|---|---| -|send|array<[OutboundMessage](#anatomy-of-an-outbound-message)>|A list of outbound messages to send when the processing of the inbound event has finished. All clients subscribed to the given channel/topic will receive the message. +|---------|----|-----------| +|send|array<[OutboundMessage](#anatomy-of-an-outbound-message)>|A list of outbound messages to send after processing the inbound event. All clients subscribed to the given channel/topic will receive the message. -##### Anatomy of an outbound message +##### Anatomy of an Outbound Message |Attribute|Type|Description| -|---|---|---| -|payload|string|The payload/body of the message you want to send. -|headers|object<string,string>|The headers/metadata of the message you want to send. -|channel|string|The channel/topic you want to send the message to. Defaults to `event.channel`, i.e., the same channel as the received event. -|server|string|The server/broker you want to send the message to. Defaults to `event.serverName`, i.e., the same server as the received event. -## How does Glee know which function it should execute? -Glee reads your `asyncapi.yaml` file and searches for all the `receive` actions containing an `operations` attribute field. The `operations` field serves as a mechanism to bind a given operation to a specific function file. For instance, given the folowing AsyncAPI definition: +|---------|----|-----------| +|payload|string|The payload/body of the message you want to send.| +|headers|object<string,string>|The headers/metadata of the message you want to send.| +|channel|string|The channel/topic to which you want to send the message. Defaults to `event.channel`, i.e., the same channel as the received event.| +|server|string|The server/broker to which you want to send the message. Defaults to `event.serverName`, i.e., the same server as the received event.| + +## How Does Glee Determine Which Function to Execute? +Glee reads your `asyncapi.yaml` file and looks for all the `receive` actions containing an `operations` attribute field. The `operations` field serves as a mechanism to bind a specific operation to a function file. For instance, given the following AsyncAPI definition: ```yaml ... operations: @@ -63,9 +64,9 @@ Glee maps the `onHello` operation to the `functions/onHello.js` file. # Lifecycle Events -Glee lets you bind incoming messages to functions. However, sometimes we need to be proactive and be the first ones to send a message, not necessarily as a reaction to another message. Use cases can be very diverse: from sending a message to announce our client is connected to sending a message every few seconds or minutes. +Glee allows you to bind incoming messages to functions. However, sometimes it's necessary to initiate communication proactively, not merely as a reaction to another message. Use cases can vary widely: from sending a message to announce that our client is connected, to broadcasting messages at regular intervals. -To subscribe to a lifecycle event, create a file under the `lifecycle` directory. It must have the following shape: +To subscribe to a lifecycle event, create a file under the `lifecycle` directory. It must be structured as follows: ```js export default async function ({ glee, @@ -79,41 +80,40 @@ export default async function ({ export const lifecycleEvent = 'onConnect' ``` -Each file in the `lifecycle` directory must export a default async function and the `lifecycleEvent` field, which is the [name of the event](#list-of-events) you want to subscribe to. Optionally, your function can return an object following exactly the same syntax as described above in the functions definition. +Each file in the `lifecycle` directory must export a default async function and the `lifecycleEvent` field, indicating the [name of the event](#list-of-events) you wish to subscribe to. Optionally, your function can return an object following the same syntax as described above in the functions definition. -## List of events +## List of Events |Event|Description| -|---|---| -|onConnect|A connection with a broker has been established. -|onReconnect|Glee reconnected to a broker. -|onDisconnect|A connection with a broker has been closed. -|onServerReady|Your Glee server is now ready to accept connections. -|onServerConnectionOpen|A client has opened a connection with your Glee server. -|onServerConnectionClose|A client has closed the connection with your Glee server. - -All of them take a single argument which contains information about the event: - -|Attribute|Description -|---|---| -|glee|A reference to the Glee app. -|serverName|The name of the server where the event happened. -|server|The AsyncAPI definition of the server where the event happened. -|connection|The connection where the event happened. - -## Restricting the lifecycle event - -In some cases it's useful to restrict the lifecycle event to a specific server or set of servers. To do that, add a line like the following to your lifecycle file: +|-----|-----------| +|onConnect|A connection with a broker has been established.| +|onReconnect|Glee reconnected to a broker.| +|onDisconnect|A connection with a broker has been closed.| +|onServerReady|Your Glee server is now ready to accept connections.| +|onServerConnectionOpen|A client has opened a connection with your Glee server.| +|onServerConnectionClose|A client has closed the connection with your Glee server.| + +All of them take a single argument containing information about the event: + +|Attribute|Description| +|---------|-----------| +|glee|A reference to the Glee app.| +|serverName|The name of the server where the event occurred.| +|server|The AsyncAPI definition of the server where the event occurred.| +|connection|The connection where the event occurred.| + +## Restricting the Lifecycle Event + +In some cases, it's useful to restrict the lifecycle event to a specific server or set of servers. To do this, add a line like the following to your lifecycle file: ```js export const servers = ['mosquitto'] ``` -The above example makes Glee fire the lifecycle event only if it's coming from the `mosquitto` server. +The above example ensures Glee fires the lifecycle event only if it originates from the `mosquitto` server. -Additionally, you may want to restrict the lifecycle event by channel/topic. To do that, add a line like the following to your lifecycle file: +Similarly, you may want to restrict the lifecycle event to a specific channel/topic. Add a line like this to your lifecycle file: ```js export const channels = ['user/signedup'] ``` -The above example makes Glee fire the lifecycle event only if the connection has the channel `user/signedup` listed as one of its channels. -Glee maps the `onHello` operation to the `functions/onHello.js` file. +The above example ensures Glee fires the lifecycle event only if the connection includes the channel `user/signedup`. diff --git a/docs/pages/glee-auth-intro.md b/docs/pages/glee-auth-intro.md index 73a136df9..8689ce36d 100644 --- a/docs/pages/glee-auth-intro.md +++ b/docs/pages/glee-auth-intro.md @@ -3,20 +3,19 @@ title: 'Introduction to Glee Authentication' weight: 60 --- -Glee comes with Authentication features which help you verifying the identity of users or entities attempting to access a system or application. It ensures that only authorised individuals or systems are granted access, protecting against unauthorised intrusions and data breaches. Glee simplifies this vital process by offering multiple authentication methods, each tailored to different use cases. Following methods are the different ways to sheild your application access. +Glee comes with authentication features that help you in verifying the identity of users or entities attempting to access a system or application. It ensures that only authorized individuals or systems are granted access, protecting against unauthorized intrusions and data breaches. Glee simplifies this vital process by offering multiple authentication methods, each tailored to different use cases. The following methods are different ways to shield your application access. ### Authentication Using Authentication Functions: -Glee allows you to implement custom authentication logic by utilising authentication functions. This flexible approach enables developers to craft tailored authentication mechanisms, ensuring that access to resources is controlled precisely as required. - +Glee allows you to implement custom authentication logic by utilizing authentication functions. This flexible approach enables developers to craft tailored authentication mechanisms, ensuring that access to resources is controlled precisely as required. ### HTTP Bearer Token Authentication: In today's API-driven world, bearer token authentication is a widely adopted method. Glee supports this approach, allowing clients to present a token as proof of their identity, thus ensuring secure and efficient access to resources. -### HttpApiKey Authentication: +### HTTP API Key Authentication: Glee's authentication suite includes support for API key authentication, which is vital for protecting web APIs. By using API keys, you can regulate access to your services, making it an essential component of your application's security strategy. ### Username and Password Authentication: Traditional yet still crucial, username and password authentication remains a reliable option within Glee's toolkit. This method allows users to access systems or applications by providing their unique credentials, ensuring a familiar and straightforward login experience. #### Summary -Glee's authentication features not only provide layers of security but also offer the flexibility needed to meet your unique requirements. Whether you're developing a web application, a mobile app, or any other application, Glee's authentication methods empower you to tailor your security measures to suit the demands of your project. With Glee, you can build and maintain a secure digital environment, ensuring that only authorised users and systems gain access, protecting your valuable data and resources. \ No newline at end of file +Glee's authentication features not only provide layers of security but also offer the flexibility needed to meet your unique requirements. Whether you're developing a web application, a mobile app, or any other type of application, Glee's authentication methods empower you to tailor your security measures to suit the demands of your project. With Glee, you can build and maintain a secure digital environment, ensuring that only authorized users and systems gain access, protecting your valuable data and resources. \ No newline at end of file diff --git a/docs/pages/glee-template.md b/docs/pages/glee-template.md deleted file mode 100644 index 8fea93cfc..000000000 --- a/docs/pages/glee-template.md +++ /dev/null @@ -1,98 +0,0 @@ ---- -title: "Create AsyncAPI Glee template" -weight: 30 ---- -This tutorial teaches you how to create a simple glee template. You'll use the AsyncAPI Glee template that you develop to generate Javascript code. Additionally, you'll create a template code with a reusable component to reuse the custom functionality you create and test your code using an WS server. - - -{`asyncapi: 3.0.0 -info: - title: 'Hello, Glee!' - version: 1.0.0 -servers: - websockets: - host: 0.0.0.0:3000 - protocol: ws -channels: - hello: - address: hello - messages: - hello: - $ref: '#/components/messages/hello' -operations: - onHello: - action: receive - channel: - $ref: '#/channels/hello' - SendHello: - action: send - channel: - $ref: "#/channels/hello" -components: - messages: - hello: - payload: - type: string`} - - -Let's break it down into pieces: - - -{`info: - title: 'Hello, Glee!' - version: 1.0.0`} - - -The `info` section provides general information about the API, including its title and version. - -Moving on, let's talk about the `servers` section. - - -{`servers: - websockets: - host: 0.0.0.0:3000 - protocol: ws`} - - -The servers section defines the different servers where the API can be accessed. In this case, there is a single server named "websockets" that uses the WebSocket protocol (`ws`) and listens on the address `ws://0.0.0.0:3000`. - -Now lets move on to the `channels` section. This section is used to describe the event names your API will be publishing and/or subscribing to. - - -{`channels: - hello: - address: hello - messages: - hello: - $ref: '#/components/messages/hello' -operations: - onHello: - action: receive - channel: - $ref: '#/channels/hello' - sendHello: - action: send - channel: - $ref: '#/channels/hello'`} - - -The channels section defines the communication channels available in the API. In this case, there's a channel named "hello". This channel supports both sending and receiving. - -The `receive` action indicates that messages received on the `hello` channel should follow the structure defined in the hello message component. -The `send` action specifies that the operation with ID `sendHello` is used for sending messages to the `hello` channel. The message structure is referenced from the hello message component. The message payload is going to be validated against the `sendHello` operation message schemas. - -Next is the `payload` property under `hello` message component which is used to understand how the event should look like when publishing to that channel: - - -{`components: - messages: - hello: - payload: - type: string`} - - -The components section contains reusable elements, in this case, a definition for the "hello" message. It specifies that the payload of the "hello" message should be of type string. - -## Summary - -In this tutorial, you learned how to create an AsyncAPI specification document via a simple example with a glee template. diff --git a/docs/pages/httpApiKey.md b/docs/pages/httpApiKey.md index ffa3dd0e8..587851f2f 100644 --- a/docs/pages/httpApiKey.md +++ b/docs/pages/httpApiKey.md @@ -3,19 +3,19 @@ title: 'HttpApiKey Authentication' weight: 90 --- -## Getting started with httpAPIKey authentication +## Getting Started with HttpAPIKey Authentication -This guide will walk through how to implement authentication using the `httpAPiKey` security scheme in Glee. +This guide will walk you through how to implement authentication using the `httpApiKey` security scheme in Glee. -A sample `asyncapi.yaml` for a server with security requirements and user `HttpApiKey` security scheme is shown below: +Below is a sample `asyncapi.yaml` for a server with security requirements and the `HttpApiKey` security scheme: ```yaml -##server asyncAPI schema +## Server AsyncAPI Schema asyncapi: 3.0.0 info: - title: AsyncAPI IMDB server + title: AsyncAPI IMDB Server version: 1.0.0 - description: This app is a dummy server that would stream the trending/upcoming anime. + description: This app is a dummy server that streams the trending/upcoming anime. servers: trendingAnimeServer: host: 'localhost:8081' @@ -37,7 +37,7 @@ components: A sample `asyncapi.yaml` for a client that implements some of the requirements of the server above: ```yaml -##client asyncAPI schema +## Client AsyncAPI Schema servers: trendingAnime: host: localhost:8081 @@ -58,38 +58,36 @@ components: type: httpApiKey name: api_key in: query - ``` -The `httpApiKey` could be in either the header or query parameter. +The `httpApiKey` can be located in either the header or query parameter. -The Client asyncapi.yaml file **does not need to implement all the security requirements in the server, it only needs to implement the ones that it uses like *httpApiKey* here.** +The client `asyncapi.yaml` file **does not need to implement all the security requirements of the server; it only needs to implement the ones it uses, like *httpApiKey* here.** ### Client Side -Following the client `asyncapi.yaml` file above, create a file named `trendingAnime.ts` in the `auth` directory, since that is the server that has the security Property. +Following the client `asyncapi.yaml` file above, create a file named `trendingAnime.ts` in the `auth` directory, as this is the server that has the security property. ```bash touch auth/trendingAnime.ts ``` -When using the `HttpApiKey` security scheme, it is important that you pass the parameters as follows: +When using the `HttpApiKey` security scheme, it is important to pass the parameters as follows: ```js -export async clientAuth({ parsedAsyncAPI, serverName }) { +export async function clientAuth({ parsedAsyncAPI, serverName }) { return { apiKey: process.env.APIKEY } } ``` -`apiKey` should be the name of the security requirement as specified in your `asyncapi.yaml` file, and it's value should be a string. +`apiKey` should be the name of the security requirement as specified in your `asyncapi.yaml` file, and its value should be a string. +### Server Side -### Server side - -From the server `asyncapi.yaml` file above, create a file named `trendingAnimeServer.ts` in the `auth` directory, since that is the server that has the security Property. +From the server `asyncapi.yaml` file above, create a file named `trendingAnimeServer.ts` in the `auth` directory, as this is the server that has the security property. ```bash touch auth/trendingAnimeServer.ts @@ -99,8 +97,8 @@ On the server side, you can retrieve the values as follows: ```js -export async serverAuth({ authProps, done }) { - authProps.getHttpAPIKeys('api_key')() +export async function serverAuth({ authProps, done }) { + authProps.getHttpAPIKeys('api_key') done(true) } diff --git a/docs/pages/index.md b/docs/pages/index.md index bc1b1d540..0f401af13 100644 --- a/docs/pages/index.md +++ b/docs/pages/index.md @@ -1,17 +1,17 @@ --- -title: Getting Started +title: Kickstarting Your Journey with Glee weight: 10 --- -## Introduction +## Welcome to Glee -[Glee](https://github.com/asyncapi/glee) is a spec-first framework that helps you build server-side applications. That means it operates on the principle of defining the API specification (AsyncAPI) before diving into the actual implementation of the application logic. It leverages that principle to make you more productive: +[Glee](https://github.com/asyncapi/glee) introduces a refreshing approach to building server-side applications, emphasizing a spec-first methodology. This means your journey begins with defining the API specification (AsyncAPI) before writing a single line of application logic. Here's how Glee enhances your development experience: -- Glee ensures your code and AsyncAPI definition are on par, eliminating the problem of outdated documentation. By having both the code and the AsyncAPI definition in sync, you can ensure that the API documentation is always up to date, accurate, and reflects the current state of the application. Glee takes care of this automatically for you. -- Glee lets you focus on what matters and handles the rest for you. You only write the code for your business use-case. Glee takes care of performance, scalability, resilience, and everything you need to make your application production-ready. -- Glee validates the schema of the payload that it receives, if it doesn't conform to the schema that is defined in the AsyncAPI document, it throw an error telling user that the server received an invalid payload. +- **Always Updated Documentation**: Glee aligns your codebase with the AsyncAPI definition, ensuring your API documentation is constantly updated and reflective of your application's current capabilities. +- **Developer Centricity**: With Glee, your focus remains on the business logic. Leave the concerns of performance, scalability, and resilience to Glee, as it equips your application to be production-ready from the get-go. +- **Schema Validation**: Glee rigorously checks incoming payloads against the schema outlined in your AsyncAPI document. Any discrepancies result in an error, maintaining the integrity and reliability of your server's data processing. -Now, before you get started with a glee project, let's take a high level view of Application structure what glee resonates with. +Before you dive into Glee, let's explore its application structure and understand what makes Glee unique. ## Application structure @@ -35,120 +35,8 @@ Glee expects your project to have some files and folders with special names. Whe |functions|**Required.** This directory contains all the functions that Glee must execute when it receives a message from the server. Each file must export a default async function. |lifecycle|This directory contains application lifecycle functions. These functions will be executed when certain events happen in the application. E.g., `onConnect`, `onServerReady`, `onDisconnect`, etc. |.env|The environment variables of your application. **DO NOT PUT SECRETS HERE**. -|asyncapi.(yaml or json or yml)|**Required.** The [AsyncAPI](https://www.asyncapi.com/docs/specifications/latest) file defines your API. Make sure all the `publish` operations have an assigned `operationId` that matches a file name (excluding the extension) in the `functions` directory. -|glee.config.js| The Glee configuration file. +|asyncapi.(yaml or json or yml)|**Required.** The [AsyncAPI](https://www.asyncapi.com/docs/specifications/latest) file defines your API. Make sure all the `receive` operations have a name that matches a file name (excluding the extension) in the `functions` directory. +|glee.config.js| The Glee [configuration file](./env-vars-config.md). |package.json|**Required.** The Node.js package definition file. Make sure you include `@asyncapi/glee` as a dependency and add two scripts: `dev` and `start`. They should be running `glee dev` and `glee start` respectively. To understand the structure in a broader way, please refer to the associated page's links. - -### Let's create a glee project to simplify the app structure - -We will consider a simple WebSocket API using glee to understand its magic. We will create a simple WebSocket server that receives a current time from the client and then send a "good morning", "good evening" or "good night" respectively. - -To setup a project, you should follow our installation page on how to setup glee on your environment. - -We recommend creating a new Glee app using our official CLI which sets up everything automatically. (You don't need to create an empty directory. create-glee-app will make one for you.) To create a project, run: `asyncapi new glee` - -Once the process is completed, you should have a new Glee app ready for development and find the files that were made. - -#### Define our Spec for our API - -Glee being a spec-first framework, development starts with defining your API spec. To know more details into it, you can follow glee template to understand it step by step. For our case we will define our API: - -```yaml -asyncapi: 3.0.0 -info: - title: Greet Bot - version: 1.0.0 -servers: - websockets: - host: 0.0.0.0:3000 - protocol: ws -channels: - greet: - address: greet - messages: - greet: - $ref: '#/components/messages/greet' - time: - $ref: '#/components/messages/time' - time: - address: time - messages: - time: - $ref: '#/components/messages/time' -operations: - onGreet: - action: receive - channel: - $ref: '#/channels/greet' - sendGreet: - action: send - channel: - $ref: '#/channels/time' -components: - messages: - time: - payload: - type: object - properties: - currentTime: - type: number - name: - type: string - greet: - payload: - type: string - -``` - -This will be the Specification that defines our API, in our case, it is very simple, as we will be sending a name and the time of the day, and our API will greet us accordingly. - -One thing to note here is the `operations` item, this is needed and is a crucial part of glee, as this is how we will be connecting our business logic with our spec, `onGreet` is the name of the function that will be called every time a certain operation occurs. In our case whenever `/greet` channel receives a message, `onGreet` function is called. - -#### Define our operation function - -Now for our case, we will be adding a file `functions/onGreet.js` and writing up the logic for parsing our time and sending a response. - -```javascript -export default async function (event) { - const { name, time } = event.payload - const t = new Date(time) - const curHr = t.getHours() - let response = '' - if (curHr < 12) { - response = `Good Morning ${name}` - } else if (curHr < 18) { - response = `Good Afternoon ${name}` - } else { - response = `Good Evening ${name}` - } - return { - send: [ - { - server: "websockets", - channel: "greet" - payload: response, - }, - ], - } -} - -``` - -Every file in the functions folder acts as a handler to develop business logic for glee, every file should export an async function that receives an event parameter, where you have access to payload and server details. - -#### Running and testing our application - -We will execute the application and carry out testing with Postman to ensure that it is functioning as intended. - -Now to execute our glee application, just run: - -``` -npm run dev -# or -npm run start -``` -To send a WebSocket request with a payload e.g. `{"name":"john", "time": "1567906535"}` to `ws://localhost:3000/greet`, open Postman and checkout the endpoint: - -So, this is how easy it is to build a WebSocket API using Glee. You can also check out the example code [here](https://github.com/Souvikns/greet-bot). diff --git a/docs/pages/installation.md b/docs/pages/installation.md index e06de1bf7..d424da085 100644 --- a/docs/pages/installation.md +++ b/docs/pages/installation.md @@ -1,42 +1,64 @@ --- -title: 'Installation guide' -weight: 20 + +title: 'Glee Installation Guide' +weight: 30 + --- -## Glee Installation +### Preparing for Installation + +Before starting with Glee, ensure that NPM, Node.js, and the [AsyncAPI CLI](https://github.com/asyncapi/cli) are already set up on your system. + +> Need help installing AsyncAPI CLI? Check out the [CLI Installation Guide](https://www.asyncapi.com/docs/tools/cli/installation). -Before installing Glee into your project, make sure you have pre-installed NPM, NodeJs and [AsyncAPI CLI](https://github.com/asyncapi/cli) tools on your system. +### Starting with Glee -### Automatic Installation +Creating a new Glee application can be approached in three distinct ways: + * [Begin with an existing AsyncAPI document](#initialize-with-an-asyncapi-document). + * [Use a predefined template](#start-with-a-template). + * [Opt for Manual Installation](#manual-installation) (recommended for advanced users keen on understanding Glee's inner workings). -The best way to get started with Glee is by using AsyncAPI CLI, which sets up everything automatically for you. -To create a project, run: +### Initialize with an AsyncAPI Document +If you already have an AsyncAPI specification file, you can jumpstart your project by using the `-f` or `--file` flag to specify the file path. The CLI will leverage this to set up your project. ```sh -asyncapi new glee +asyncapi new glee -f asyncapi.yaml ``` -> For more information on how to install the AsynAPI CLI, you can review the [CLI installation guide](https://www.asyncapi.com/docs/tools/cli/installation). +This command generates all necessary files. Just open the project in your editor, and start integrating your authentication and business logic. + +### Start with a Template + +For those without a custom AsyncAPI file and looking to experiment, starting with a template is an excellent choice. These are simple Glee projects that the AsyncAPI CLI can generate for you. + +Use the `-t` or `--template` flag to specify the template name. Currently, two templates are available: -On installation, you'll find next steps after your project created: +1) **default**: Responds with a string upon receiving a string. +2) **tutorial**: Designed for [this tutorial](https://www.asyncapi.com/docs/tutorials/generate-code). +To initiate a project with the default template: + +```sh +asyncapi new glee -t default ``` -Your project "project" has been created successfully! +After installation, follow these steps: + +``` Next steps: cd project - npm install + npm install --ignore-scripts npm run dev -Also, you can already open the project in your favorite editor and start tweaking it +Feel free to open the project in your preferred editor and start customizing it. ``` -While making twists to your application, you can follow along with our getting started guide on the relevant page. +Refer to our Getting Started guide for further assistance during customization. ### Manual Installation -To manually create a new app, create a new folder e.g. `myapp` so the folder structure would look like below; +For a hands-on setup, start by creating a new directory, e.g., `myapp`, and structure it as follows: ``` ├─ functions (required) @@ -51,14 +73,14 @@ To manually create a new app, create a new folder e.g. `myapp` so the folder str ├─ package.json (required) ``` -Install the required packages inside a new folder: +Inside this new folder, initialize and install Glee: ```js npm init -y npm install @asyncapi/glee ``` -Open your package.json file and add the following scripts: +Edit your `package.json` to include the following scripts: ```js { @@ -70,19 +92,18 @@ Open your package.json file and add the following scripts: } ``` -These scripts refer to the different stages of developing an application. - -- `glee docs`: This script generates documentation for your project using the "Glee" documentation tool. This documentation includes information about your project's APIs, modules, and usage instructions. - -- `glee dev`: This script is used for starting a development server. It launches a local development server, build your project in development mode, or perform other development-related tasks. +These scripts serve different development stages: -- `glee start`: This script is responsible for starting your project or application. It is used to launch a production-ready server or application instance. +- `glee docs`: Generates your project documentation. +- `glee dev`: Starts a local development server, building your project in a development-friendly environment. +- `glee start`: Launches your project for production use. -#### Create `asyncapi.yaml` file and other required directories +#### Setting up `asyncapi.yaml` and Required Directories -Create a yaml file that supports capable of receiving a "hello {name}" message with the protocol as `ws` and the channel name as `hello` the hello API will subscribe to. The operationId property is `onHello` that's the name of function and the payload property is type string publishing to that channel. +Craft an `asyncapi.yaml` file capable of receiving a "hello {name}" message via WebSocket (`ws`) protocol on the `hello` channel. Define the operation ID as `onHello`, indicating the function to be called, and set the payload type to string for publishing on that channel. ```yaml +# AsyncAPI Specification for Hello, Glee! asyncapi: 3.0.0 info: title: 'Hello, Glee!' @@ -113,7 +134,7 @@ components: type: string ``` -Create an operation function `onHello.js` inside `myapp/functions`: +Create a function `onHello.js` in `myapp/functions`: ```js export default async function (event) { @@ -127,7 +148,7 @@ export default async function (event) { } ``` -#### Run the Development Server +#### Launching the Development Server -- Run `npm run dev` to start the development server. -- Connect to `ws://localhost:3000/hello` and send a WebSocket request with a payload e.g. {"john"} +- Execute `npm run dev` to start the server. +- Connect to `ws://localhost:3000/hello` and send a WebSocket request like `{"john"}`. \ No newline at end of file diff --git a/docs/pages/your-first-glee-app.md b/docs/pages/your-first-glee-app.md new file mode 100644 index 000000000..dfdc3051f --- /dev/null +++ b/docs/pages/your-first-glee-app.md @@ -0,0 +1,125 @@ +--- + +title: Building Your First Glee Application +weight: 20 + +--- + +#### Crafting Your API Specification + +Let's start with a straightforward WebSocket API using Glee. Imagine a WebSocket server that receives the current time from a client and responds with a greeting like "good morning", "good evening", or "good night" based on the time provided. + +In Glee, which is a spec-first framework, the journey begins with defining your API specification. If you're unfamiliar with what an API spec is or how to create one, we recommend going through the tutorials available [here](https://www.asyncapi.com/docs/tutorials) before proceeding. For this project, here's how we define our API: + +```yaml +asyncapi: 3.0.0 +info: + title: Greet Bot + version: 1.0.0 +servers: + websockets: + host: localhost:3000 + protocol: ws +channels: + greet: + address: /greet + messages: + greet: + payload: + type: string + time: + payload: + type: object + properties: + currentTime: + type: number + name: + type: string +operations: + receiveTime: + action: receive + channel: + $ref: '#/channels/greet' + messages: + - $ref: "#/channels/greet/messages/time" + sendGreet: + action: send + channel: + $ref: '#/channels/greet' + messages: + - $ref: "#/channels/greet/messages/greet" +``` + +Store this in a file named `asyncapi.yml`. + +This spec shows that our app can perform two operations on a single channel: receiving time and sending a greeting. We've used JSON Schema to define what `time` and `greet` messages look like. + +Note the `operations` section. It's essential in Glee, linking your business logic to your spec. For instance, `receiveTime` is the function invoked when the `/greet` channel receives a message. + +> Tip: Including a `send` operation isn't mandatory but is recommended for validating outgoing messages against your spec. + +### Initiating Your Glee Project + +For ease and efficiency, start your Glee app with our CLI, which automates the setup. In your `asyncapi.yml` file's directory, execute: `asyncapi new glee --name GreetBot --file asyncapi.yml` + +It will prompt you with a list of server names and let you select which servers you want Glee to create a server for. select the `websockets` and continue. + +Then, navigate to your application's folder and install the dependencies: + +```shell +cd GreetBot +npm install --ignore-scripts +``` + +You now have a Glee app scaffolded and ready for development. + +#### Implementing the Operation Function + +Navigate to `functions/receiveTime.js` and input the logic to analyze the time and generate the appropriate response. + +```ts +import { GleeFunction } from '@asyncapi/glee'; + +const receiveTime: GleeFunction = async (event) => { + const { name, time } = event.payload; + const t = new Date(time); + const curHr = t.getHours(); + let response = ''; + if (curHr < 12) { + response = `Good Morning, ${name}!`; + } else if (curHr < 18) { + response = `Good Afternoon, ${name}!`; + } else { + response = `Good Evening, ${name}!`; + } + return { + send: [ + { + server: 'websockets', + channel: 'greet', + payload: response, + }, + ], + }; +}; + +export default receiveTime; +``` + +> Note: Want to know more about functions? [Click Here!](./function-lifecycle-events.md) + +Each file in the `functions` directory is a handler where you can craft your business logic. Every handler should export an asynchronous function that takes an `event` parameter, giving you access to the payload and server details. + +#### Running and Evaluating Your App + +To launch and test the app's functionality, follow these steps: + +Run your Glee application using: + +``` +npm run dev +# or +npm run start +``` + +To test, open a WebSocket connection to `ws://localhost:3000/greet` in Postman and send `{"name":"John","time":"1567906535"}`. Watch as your Glee app responds appropriately. \ No newline at end of file