https://github.com/lab34-es/lab34-flows
Test automation for local and remote environments with AI capabilities
https://github.com/lab34-es/lab34-flows
test-automation testing testing-tools
Last synced: 3 months ago
JSON representation
Test automation for local and remote environments with AI capabilities
- Host: GitHub
- URL: https://github.com/lab34-es/lab34-flows
- Owner: lab34-es
- Created: 2025-09-19T09:41:53.000Z (10 months ago)
- Default Branch: master
- Last Pushed: 2025-09-24T18:10:30.000Z (9 months ago)
- Last Synced: 2025-10-24T04:25:34.201Z (8 months ago)
- Topics: test-automation, testing, testing-tools
- Language: JavaScript
- Homepage:
- Size: 351 KB
- Stars: 0
- Watchers: 0
- Forks: 0
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
README
# Flows
Heavily opinionated tool to help you test E2E flows and behaviours.
> Trigger, understand and test E2E flows and behaviours.
> Ask an AI to generate a flow for you.
> Run on any environment (staging, production, etc).
> From Async protocols, APIs and databases, to web applications.
Features:
- Integrates with Google Gemini AI to generate test scenarios.
- Define flows in YAML format that can be shared across members and teams. (no need to explain failing cases verbally).
- Generate flows automatically using AI with natural language prompts.
- Define test cases for each step in a way that can be used for CI/CD automation.
- Mimic and customise the behaviour of dependant applications. (i.e. fail scenarios under user defined circusnstances).
- Test with randomly generated data - on each attempt.
- Integrates with MQTT, HTTP apis, PostgreSQL databases and web applications.
- Keeps environment variables in a safe place.
- Offers a comprehensive view of the flowing information (headers, body, etc).
- Built-in replacers for customising requests and responses.
- Retry / delay mechanism for failed steps.
- Test flows locally, mimicing dependent applications, without leaving the comform of your localhost.
## Table of contents
- [Flows tools](#flows-tool)
- [Table of contents](#table-of-contents)
- [General info](#general-info)
- [Setup](#setup)
- [Usage](#usage)
- [AI Mode](#ai-mode)
- [Flows](#flows)
- [Tests](#tests)
- [Playwright](#playwright) (browser automation - experimental)
- [Replacers](#replacers)
- [Environment variables](#environment-variables)
## General info
This tool is intended to help you test E2E flows and behaviours.
The list of steps to execute are defined is files called "flows".
This files are written in YAML format, and are located in the `flows` folder in this repository. Though, you can have your own flows in your own computer / repository.
The tool will read the flows, and execute the steps in the order they are defined.
Additionally:
1. You can mimic the behaviour of dependant applications, by adding a *mimic* section to each step's definition.
2. You can define test cases for each step, and the tool will execute them.
En example of a flow file is:
```yaml
steps:
- application: "accounting"
method: "getInvoice"
parameters:
params:
customerId: "{{ randomInt0_100 }}"
query:
from: "2023-01-01"
to: "2023-01-31"
headers:
X-Vendor-Name: "ACME"
mimic:
- application: "marosavat"
url:
- "taxes"
- application: "coinscrap"
url: "/frau-detection"
conditions:
fraudForCustomer:
- "57"
test:
status: 404
headers:
- Content-Type: "application/json"
body:
error:
httpStatusCode: 404
code: "ACCOUNTING_FRAUD_DETECTED"
message: "Fraud detected in accounting for customer 57"
```
## Setup
```bash
# Install this repository, globally.
npm install -g @lab34/flows
# Extend NODE_PATH to npm's root, so your application scripts can access the library:
# On Linux / MacOS:
export NODE_PATH=$(npm root -g)
# On Windows:
set NODE_PATH=%AppData%\npm\node_modules
# On NVM for Windows:
set NODE_PATH=%NVM_SYMLINK%\node_modules
# On NVM for Windows (powershell)
$env:NODE_PATH = "$env:NVM_SYMLINK\node_modules"
```
## Usage
The Lab34 Flows CLI tool provides a professional command-line interface for running flow definitions from YAML files.
### Usage
```bash
lab34-flows --help
lab34-flows --file --env [--debug] [--help]
lab34-flows --capabilities
lab34-flows --ai ""
```
### Options
|Parameter|Description|
|-|-|
|`--file`|Path to the YAML flow definition file (required if not using --ai)|
|`--ai`|Generate a flow from a prompt using AI (required if not using --file)|
|`--env`|Environment to run the flow in (required for --file, optional for --ai)|
|`--debug`|Print debug information including environment variables and Node.js variables|
|`--help`|Show help information|
### Examples
Display help information:
```bash
lab34-flows --help
```
Run a flow with debug information:
```bash
lab34-flows --file flows/my-flow.yaml --env production --debug
```
Generate and run a flow using AI:
```bash
lab34-flows --ai "Test login functionality with valid credentials"
```
### AI Mode
The AI mode allows you to generate flow definitions using natural language prompts. This feature leverages Google's Generative AI (Gemini) to create YAML flow definitions based on your description of the testing scenario.
#### AI Configuration
Before using the AI feature, you need to set up your AI configuration:
1. Create a file named `ai.json` in the `~/flows/config/` directory with the following structure:
```json
{
"defaultProvider": "gemini",
"gemini": {
"apiKey": "YOUR_GEMINI_API_KEY_HERE",
"model": "gemini-pro",
"temperature": 0.7,
"topP": 0.95,
"topK": 40
}
}
```
2. Replace `YOUR_GEMINI_API_KEY_HERE` with your actual Gemini API key.
#### Using AI Mode
Once configured, you can use the AI mode to generate flows:
```bash
lab34-flows --ai "Test the user registration process with valid data"
```
The tool will:
1. Send your prompt to the AI service
2. Generate a YAML flow definition based on your description
This feature is particularly useful for:
- Quickly creating test flows without manually writing YAML
- Exploring different testing scenarios
- Generating comprehensive test cases from simple descriptions
### Debug Mode
When the `--debug` flag is used, the CLI tool will print detailed information about:
1. All environment variables
2. Node.js variables including:
- `__dirname`: The directory name of the current module
- `__filename`: The file name of the current module
- `process.cwd()`: The current working directory
- `process.argv`: The command line arguments
This information is useful for troubleshooting and understanding the execution environment.
## Replacers
Replacers are used to customize the requests and responses of the steps in the flow, as well as the mimicked applications. The tool uses Handlebars templates for replacements.
For example, you can define a value like `{{ randomInt0_100 }}` in the flow file, or in the mimicked application responses, and the tool will replace it with a randomly generated integer between 0 and 100.
### Basic Replacers
The following basic replacers are available:
| Replacer | Description | Example |
|-------------------|--------------------------------------------------|----------------------------------|
| `timestamp` | Current timestamp in milliseconds | `1633024800000` |
| `datetime` | Current date and time in ISO format | `2023-10-01T12:00:00.000Z` |
| `randomInt` | Random integer between 0 and 999 | `42` |
| `randomInt0_5` | Random integer between 0 and 4 | `3` |
| `randomInt0_10` | Random integer between 0 and 9 | `7` |
| `randomInt0_100` | Random integer between 0 and 99 | `56` |
| `randomInt0_200` | Random integer between 0 and 199 | `123` |
| `randomInt0_300` | Random integer between 0 and 299 | `250` |
| `randomInt0_500` | Random integer between 0 and 499 | `400` |
| `randomInt0_1000` | Random integer between 0 and 999 | `789` |
| `randomInt0_9999` | Random integer between 0 and 9998 | `6789` |
| `randomInt0_2000` | Random integer between 0 and 1999 | `1500` |
| `randomInt0_3000` | Random integer between 0 and 2999 | `2500` |
| `randomInt0_4000` | Random integer between 0 and 3999 | `3500` |
| `randomInt0_5000` | Random integer between 0 and 4999 | `4500` |
| `uuid` | Random UUID | `123e4567-e89b-12d3-a456-426614174000` |
| `randomPostmanId` | Random 6-digit integer | `123456` |
### Personal Data Replacers
| Replacer | Description | Example |
|-----------------------|----------------------------------------------|----------------------------------|
| `randomEmail` | Randomly generated email address | `user123@example.com` |
| `randomName` | Random person full name | `John Doe` |
| `randomPersonName` | Random person first name | `Jane` |
| `randomPersonSurname` | Random person last name | `Smith` |
| `randomPersonPrefix` | Random person name prefix | `Mr.` |
| `phoneIntl` | Random phone number in international format | `+1 555-123-4567` |
| `randomString` | Random alphanumeric string (10 characters) | `a1b2c3d4e5` |
### Location and Company Replacers
| Replacer | Description | Example |
|-----------------------|----------------------------------------------|----------------------------------|
| `belgianCityEn` | Random Belgian city name in English | `Brussels` |
| `randomCompanyName` | Random company name | `Acme Corporation` |
| `randomStreet` | Random street name | `Main Street` |
| `randomStreetNumber` | Random street number between 0 and 199 | `42` |
| `randomPostalCode` | Random 4-digit postal code | `1000` |
### Time-Related Functions
You can generate dates and timestamps in the past using the following helper functions:
```text
timeAgo amount lapse
timestampAgo amount lapse
tsAgo amount lapse
```
Where:
- `amount`: The number of time units to go back
- `lapse`: The time unit (ms, seconds, minutes, hours, days, months, years)
Examples:
```text
timeAgo 5 "days"
timestampAgo 2 "hours"
tsAgo 1 "month"
```
### Barcode Generation
You can generate random barcodes using the barcode helper:
```javascript
barcode([pattern])
```
Where:
- `pattern`: is an array of strings and/or numbers.
- An string is a fixed value.
- A number adds N number of numbers to the barcode.
Examples:
```javascript
barcode(["123456", 3, "789"]) // Generates a barcode like "123456123789"
barcode(["HELLO-", 4, "-WORLD"]) // Generates a barcode like "HELLO-7832-WORLD"
```
### Random Selection
You can select a random element from an array using the oneOf helper:
```javascript
oneOf([array])
```
You can contribute and add more replacers by modifying the `src/helpers/replacer.js` file.
## Flows
Flows are defined in YAML format. Basic examples are available in the `flows` folder of the repository.
Though, you can have your own flows in your own computer / repository, and share them with your team.
## Tests
Tests are defined in the flow file, and are executed by the tool.
You can test two aspects on each step:
1. The it has returned the expected status code and body. (i.e. 200, 404, etc, with certain body contents).
2. Lantent applications. See [Latent Applications](#latent-applications) for more information.
### JavaScript Expressions in Tests
You can use JavaScript expressions in your test definitions to perform dynamic validations beyond simple equality checks. To use an expression, prefix it with `$expr:` followed by valid JavaScript code, where `value` represents the actual value being tested.
Examples:
```yaml
test:
body:
count: "$expr: value > 10" # Validates that count is greater than 10
status: "$expr: value === 'completed'" # Validates that status equals 'completed'
items: "$expr: Array.isArray(value) && value.length >= 3" # Validates items is an array with at least 3 elements
user:
age: "$expr: value >= 18 && value <= 65" # Validates age is between 18 and 65
timestamp: "$expr: new Date(value).getFullYear() === 2023" # Validates year is 2023
```
Common validation scenarios:
| Validation Type | Expression Example |
|----------------|-------------------|
| Greater than | `$expr: value > 0` |
| Equals specific value | `$expr: value === 2` |
| In a range | `$expr: value >= 5 && value <= 10` |
| String contains | `$expr: typeof value === 'string' && value.includes('success')` |
| Array has items | `$expr: Array.isArray(value) && value.length > 0` |
| Property exists | `$expr: typeof value === 'object' && 'id' in value` |
| Date validation | `$expr: new Date(value) > new Date('2023-01-01')` |
This feature allows for powerful and flexible test assertions without having to modify the testing code.
## Latent Applications
**Only works for MQTT at the moment**
This are applications that are triggered asynchonously by some step in the flow, for which test validation is important in order to consider a flow as successful.
For example: you might want to validate the, upon triggering a HTTP request, a certain MQTT has been produced at some point.
To do so, you can define this applications in the `latentApplications` section in the flow.
```yaml
latentApplications:
- application: "mqtt"
client: "client1"
connection:
host: "1234567890-ats.iot.eu-west-1.amazonaws.com"
key: "/Users/myuser/mqtt-credentials/private.key"
cert: "/Users/myuser/mqtt-credentials/cert.crt"
ca: "/Users/myuser/mqtt-credentials/ca1.pem"
subscribe:
- topic: "client/1"
steps:
...
```
In this example, we are ensuring we will have a MQTT client connected to the given host, and subscribed to a list of topics, before the flow starts.
This enables the flow to be able to test that the MQTT messages are being produced as expected.
For example, in the following example, we are ensuring that a MQTT message containing `status = switched_to_on` is received in the `client/1` topic by the client described above.
```yaml
steps:
- application: ""
method: ""
test:
latentApplications:
- application: "mqtt"
client: "client1"
test:
- topic: "client/1"
message:
status: "switched_to_on"
retry:
attempts: 1
delay: 1
```
## Playwright
Testing web applications is possible with the tool, as it integrates with [Playwright](https://playwright.dev).
Playwright automations have their own YAML files, as you can see in [an actual example](src/applications/playful_website/playwright.example.yaml).
To integrate with Playwright, the application's code must make use of `playwright.run` and pass a playwright yaml file to it.
### Configuration Options
The Playwright integration now supports enhanced configuration through the YAML file:
```yaml
# Browser configuration
browserType: chromium # Options: chromium, firefox, webkit
device: Desktop Chrome # Any device from playwright.devices
keepOpen: true # Keep the browser open after execution (only for debugging purposes)
# Launch options
launchOptions:
headless: false
ignoreHTTPSErrors: true
timeout: 30000
args: [] # Additional browser arguments
# Context options
contextOptions:
locale: en-US
timezoneId: Europe/Madrid
permissions: []
geolocation: null
viewport:
width: 1280
height: 720
```
### Available Methods
|Method|Description|Parameters|
|-|-|-|
|goto|Navigate to a URL|`url`, `waitUntil`, `timeout`|
|click|Clicks in a element|`selector`, `button`, `clickCount`, `delay`, `timeout`|
|type|Types a text in a form field|`selector`, `text`, `delay`, `timeout`|
|fill|Fill a form field|`selector`, `value`, `timeout`|
|press|Press a key|`selector`, `key`, `delay`, `timeout`|
|hover|Hover over an element|`selector`, `position`, `timeout`|
|dragAndDrop|Drag and drop operation|`source`, `target`, `force`, `timeout`|
|selectOption|Select dropdown option|`selector`, `values`, `timeout`|
|check|Check a checkbox|`selector`, `position`, `timeout`|
|uncheck|Uncheck a checkbox|`selector`, `position`, `timeout`|
|evaluate|Execute JavaScript|`pageFunction`, `arg`|
|keyboard|Keyboard actions|`action`, `args`|
|mouse|Mouse actions|`action`, `args`|
|waitForTimeout|Wait for time|`time`|
|waitForSelector|Wait for element|`selector`|
|assertTitle|Assert page title|`title`|
|screenshot|Take screenshot|`path`|
|waitForInput|Wait for user input|-|
|scrape|Extract page data|`selector`, `output`|
### Method Examples
#### goto
```yaml
method: goto
parameters:
url: "https://example.com"
waitUntil: "networkidle" # Options: load, domcontentloaded, networkidle
timeout: 30000
```
#### click
```yaml
method: click
parameters:
selector: "#submit-button"
button: "left" # Options: left, right, middle
clickCount: 1
delay: 100
timeout: 5000
```
#### type
```yaml
method: type
parameters:
selector: "#search"
text: "Búsqueda de ejemplo"
delay: 50 # Milisegundos entre pulsaciones
timeout: 5000
```
#### fill
```yaml
method: fill
parameters:
selector: "#username"
value: "user123"
timeout: 5000
```
#### press
```yaml
method: press
parameters:
selector: "#search-input"
key: "Enter"
delay: 100
timeout: 5000
```
#### hover
```yaml
method: hover
parameters:
selector: ".dropdown-menu"
position: { x: 0, y: 0 } # Coordenadas relativas al elemento
timeout: 5000
```
#### dragAndDrop
```yaml
method: dragAndDrop
parameters:
source: "#draggable"
target: "#droppable"
force: true
timeout: 5000
```
#### selectOption
```yaml
method: selectOption
parameters:
selector: "#country-select"
values: ["ES"] # Puede ser un string único o un array
timeout: 5000
```
#### check
```yaml
method: check
parameters:
selector: "#terms-checkbox"
position: { x: 5, y: 5 } # Opcional: coordenadas específicas para el click
timeout: 5000
```
#### uncheck
```yaml
method: uncheck
parameters:
selector: "#newsletter-checkbox"
position: { x: 5, y: 5 }
timeout: 5000
```
#### dblclick
```yaml
method: dblclick
parameters:
selector: "#edit-field"
timeout: 5000
```
#### focus
```yaml
method: focus
parameters:
selector: "#email-input"
timeout: 5000
```
#### evaluate
```yaml
method: evaluate
parameters:
pageFunction: "() => document.title"
arg: null
```
#### keyboard
```yaml
method: keyboard
parameters:
action: "type"
args: ["Hello World", { delay: 100 }]
```
#### mouse
```yaml
method: mouse
parameters:
action: "move"
args: [100, 200] # x, y coordinates
```
#### waitForTimeout
```yaml
method: waitForTimeout
parameters:
time: 1000 # Milisegundos
```
#### waitForSelector
```yaml
method: waitForSelector
parameters:
selector: ".loading-indicator"
state: "hidden" # Options: 'attached', 'detached', 'visible', 'hidden'
timeout: 30000
```
#### assertTitle
```yaml
method: assertTitle
parameters:
title: "Página de inicio"
```
#### screenshot
```yaml
method: screenshot
parameters:
path: "screenshots/error-state.png"
fullPage: true # Opcional: captura toda la página
omitBackground: false # Opcional: fondo transparente
```
#### waitForInput
```yaml
method: waitForInput
# No requiere parámetros - espera entrada del usuario
```
#### scrape
```yaml
method: scrape
parameters:
title:
selector: "h1"
output: "string"
price:
selector: ".price"
output: "number"
isAvailable:
selector: ".stock-status"
output: "boolean"
publishDate:
selector: ".publish-date"
output: "date"
```
## Environment variables
To prevent storing credentials and others in the repository, the testers must keep environment files for each supported application in the application folder itself.
i.e. for an application called ABC, the configuration files must be stored in `~/.../applications/abc/envs/ac1.env`.
### PostgreSQL Database Configuration
The PostgreSQL client supports flexible configuration through environment variables. You can either use a connection string or individual parameters.
#### Option 1: Connection String (Recommended for simplicity)
```bash
DATABASE_CONNECTION_STRING=postgres://user:password@host:5432/database
```
#### Option 2: Individual Parameters (Recommended for flexibility)
| Environment Variable | Description | Example |
|---------------------|-------------|---------|
| `PGUSER` | Database user | `myuser` |
| `PGPASSWORD` | Database password | `mypassword` |
| `PGHOST` | Database host | `localhost` or `db.example.com` |
| `PGPORT` | Database port | `5432` |
| `PGDATABASE` | Database name | `mydatabase` |
| `PGQUERY_TIMEOUT` | Query timeout in milliseconds | `30000` |
| `PGLOCK_TIMEOUT` | Lock timeout in milliseconds | `10000` |
| `PGCLIENT_ENCODING` | Client character encoding | `UTF8` |
| `PGOPTIONS` | Command-line options for the server | `-c statement_timeout=30s` |
#### Configuration Priority
1. If `DATABASE_CONNECTION_STRING` is provided, it takes precedence (for backward compatibility)
2. Otherwise, individual parameters (`PGUSER`, `PGHOST`, etc.) are used
3. Additional parameters (`PGQUERY_TIMEOUT`, `PGLOCK_TIMEOUT`, etc.) work with both approaches
#### SSL Configuration (Optional)
The PostgreSQL client supports SSL connections with flexible configuration options:
| Environment Variable | Description | Example |
|---------------------|-------------|---------|
| `PGSSL_ENABLED` | Enable SSL connection | `true` or `false` |
| `PGSSL_REJECT_UNAUTHORIZED` | Reject unauthorized certificates | `true` or `false` |
| `PGSSL_CA` | Path to CA certificate file | `/path/to/ca.pem` |
| `PGSSL_CERT` | Path to client certificate file | `/path/to/client-cert.pem` |
| `PGSSL_KEY` | Path to client key file | `/path/to/client-key.pem` |
#### Examples
**Using connection string:**
```bash
DATABASE_CONNECTION_STRING=postgres://admin:secret@db.example.com:5432/production
PGQUERY_TIMEOUT=60000
```
**Using individual parameters:**
```bash
PGUSER=admin
PGPASSWORD=secret
PGHOST=db.example.com
PGPORT=5432
PGDATABASE=production
PGQUERY_TIMEOUT=60000
PGLOCK_TIMEOUT=30000
```
**Local development example:**
```bash
PGUSER=developer
PGPASSWORD=localpass
PGHOST=localhost
PGPORT=5432
PGDATABASE=testdb
```
**SSL connection with self-signed certificates (development):**
```bash
PGUSER=developer
PGPASSWORD=devpass
PGHOST=secure.dev.example.com
PGPORT=5432
PGDATABASE=devdb
PGSSL_ENABLED=true
PGSSL_REJECT_UNAUTHORIZED=false
```
**SSL connection with full certificate validation (production):**
```bash
DATABASE_CONNECTION_STRING=postgres://admin:secret@secure.db.example.com:5432/production
PGSSL_ENABLED=true
PGSSL_CA=/path/to/ca-certificate.pem
PGSSL_CERT=/path/to/client-certificate.pem
PGSSL_KEY=/path/to/client-key.pem
```
**SSL with custom CA certificate only:**
```bash
PGUSER=admin
PGPASSWORD=secret
PGHOST=db.example.com
PGPORT=5432
PGDATABASE=production
PGSSL_ENABLED=true
PGSSL_CA=/path/to/custom-ca.pem
```