Cloud
Studio Interface
Tables

Tables

Studio Explorer

Tables are an essential feature in the explorer, functioning as a local database within your chatbot's environment. They provide structured storage for various types of data that can be accessed and manipulated as required within your workflows.

They are particularly useful for storing information that needs to be persisted across multiple sessions, such as user profiles, transaction histories, and other types of data that need to be retrieved and updated over time.

Note
You can create up to 10 tables with 20 columns and 10,000 rows in the Free Plan.
Get in touch (opens in a new tab) if you want to increase the limits. However, the 20 columns per table is a hard limit that can't be increased.

Creating a Table

To add a new table, go to the explorer menu and select Add Table. You can then give your table a name and add fields to it. Each field has a name and a type. The type determines the kind of data that can be stored in the field. For example, a field with the type String can store any text value, while a field with the type Number can only store numeric values.

Making Fields Searchable

You can make a field searchable by checking the Searchable checkbox. This will allow you to search for records based on the values in the field. For example, if you have a field called Name and you make it searchable, you can search for records based on the name of the person. This is only available for fields with the type String.

Types of Fields (Columns)

There are several types of fields you can add to a table. Each field type has a specific purpose and can only store certain types of data. These are the field types you can choose from: String, Number, Boolean, Date, and Object.

⚠️

The number of columns has a hard limit of 20 columns per table.

Query Selector

A Query Selector is the query you use to select or filter data from your table. For instance, you may need to find a specific record based on certain criteria like a user's ID or a transaction number. The Query Selector allows you to articulate these conditions in plain English, making it both intuitive and powerful. An example of a query could be "Find all records where 'user_id' equals '123'".

To add a Query Selector, click on the table in the explorer menu. You'll then see a list of all the records in the table. Click on the Add Query Selector button from the right panel to add a new query selector. You can then enter the query selector in the text field.

Filtering Records

Table filters are a powerful feature that allows you to refine and display only the records that meet certain criteria. Filters can be based on specific conditions, such as date ranges, text matches, and numerical comparisons.

Logical Operators

  • AND: Use the AND operator when you want all conditions to be true for a record to be displayed. For example, filtering by createdAt AND status will only show records that match both conditions.

  • OR: Use the OR operator to display records that match any one of multiple conditions. For example, filtering by createdAt OR updatedAt will show records that match either one of the conditions.

Types of Rules

Rules define the individual conditions applied to each column of data.

  • is equal to: Shows records where the field is equal to the specified value.
  • is not equal to: Shows records where the field is not equal to the specified value.
  • less than: Shows records where the field is less than the specified value.
  • less than or equal to: Shows records where the field is less than or equal to the specified value.
  • greater than: Shows records where the field is greater than the specified value.
  • greater than or equal to: Shows records where the field is greater than or equal to the specified value.
  • in: Shows records where the field matches any of the specified values.
  • not in: Shows records where the field does not match any of the specified values.
  • is null: Shows records where the field is empty.
  • is not null: Shows records where the field is not empty.

Rule Groups

  • Group: A group is a collection of rules combined by a logical operator (AND/OR). You can nest groups to create complex filters.

Creating a Filter

  1. Click on the Filter button to expand the filter menu.
  2. Select a logical operator (AND/OR) from the dropdown.
  3. Choose a field to filter by (e.g., createdAt).
  4. Select a rule (e.g., is equal to).
  5. Enter the required value or select from a calendar if it's a date field.
  6. To add another rule within the same logical condition, click the + Rule button.
  7. To create a nested group of conditions, click the + Group button.
  8. Apply the filter by clicking the Apply filter button.

Manually Editing Filters

Click on Edit Manually to view and edit the filter in JSON format. You can then copy this JSON filter and use it in your table operations via code.

Operations on Tables - via the Interface

Add records to a table

Once you've created a table, you can add records to it. To do this, click on the table in the explorer menu. You'll then see a list of all the records in the table. Click on the Add Record button to add a new record. You can then enter the values for each field in the record.

Update records in a table

Double click on a a table cell or press click it and press Enter to see an input that allows you to change the cell value. Once you are done, press Enter again or click away to save the new value.

Delete records from a table

Right Click on the Record and select Delete Record to delete a record from a table.

Operations on Tables - via Cards

Check out the Table Cards documentation to learn more about operations with cards.

Operations on Tables - via Botpress Client

You can also interact with tables using the Botpress Client. Here are some of the functions you can use to interact with tables:

listTables

To list all the tables in the bot, use the listTables function. This function will return an array of all the tables in the bot.

try {
  const tables = await client.listTables({})
  console.log('Tables:', tables)
} catch (err) {
  console.error('Error fetching tables', err)
}

Create Table Rows

Execute the following query to create one or multiple rows. The response will include the inserted rows, along with any encountered errors or warnings during the operation (e.g., unrecognized fields)

const { rows, errors, warnings } = await client.createTableRows({
  table: 'Data1Table',
  rows: [
    {
      name: 'test'
    }
  ]
})

Find & Filter Table Rows

To find and filter rows in a table, use the findTableRows function. You can specify the limit (number of records to return) and the offset (number of rows to start from).

const { rows, limit, offset, count } = await client.findTableRows({
  table: 'Data1Table',
  limit: 50,
  offset: 0,
  filter: {},
  orderBy: 'row_id',
  orderDirection: 'asc'
})

Utilize filters to conduct more refined queries for specific rows, employing a MongoDB-like syntax. The query builder aids in understanding available operations and constructing the filter.

const { rows, limit, offset, count } = await client.findTableRows({
table: 'Data1Table',
filter: {
  // Retrieve rows where date is greater than a specific date
  updatedAt: { gte: '2024-01-01' },
  // Or more complex conditions, such as name starting with 'a' or 'b', or 'isActive' being true
  $or: [{ name: { $regex: '^a' } }, { name: { $regex: '^b' } }, { isActive: true }]
}
})

The "search" property allows for a partial text search across the table. The query returns multiple results, each accompanied by a similarity score to indicate relevance.

const { rows, limit, offset, count } = await client.findTableRows({
  table: 'Data1Table',
  search: 'John Doe'
})

Projection & Aggregation

Employ the group property for projections and aggregations. The following example demonstrates returning a single row with requested data.

const { rows } = await client.findTableRows({
  table: 'Data1Table',
  group: {
    id: 'count',
    name: ['unique', 'count'], // will return a list of unique values & the total count
    updatedAt: ['min', 'max'] // Retrieves the earliest and latest date
  }
})
 
// Output variables are named after the field, suffixed with the operation
const oldestChange = rows[0].updatedAtMin
const mostRecentChange = rows[0].updatedAtMax
const count = rows[0]['nameCount']
const uniqueValues = rows[0]['nameUnique']
ColumnTypeKeyCountSumAvgMaxMinUnique
idnumber✔️✔️✔️✔️✔️✔️✔️
createdAtdate✔️✔️✔️✔️
updatedAtdate✔️✔️✔️✔️
namestring✔️✔️✔️✔️
  • Key: Projections are grouped by keys (ex: date)
  • Count: Returns the total number of non-null values
  • sum: Returns the sum of all values
  • Avg: Returns the average of all values
  • Max: Returns the maximum value
  • Min: Returns the minimum value
  • Unique: Returns a list of unique values

Providing one or multiple keys yields more specific results.

const { rows } = await client.findTableRows({
  table: 'Data1Table',
  group: {
    someField: 'key', // Either provide a single operation
    someNumber: ['avg', 'sum'] // Or multiple operations
  }
})
 
// Returns multiple rows for each unique combination of keys
const result = rows.map((row) => {
  const { someFieldKey, someNumberAvg, someNumberSum } = row
})

Update Table Rows

This operation updates rows based solely on their numeric IDs.

const { rows, errors, warnings } = await client.updateTableRows({
  table: 'Data1Table',
  rows: [
    {
      id: 1,
      name: 'test'
    }
  ]
})

Delete Table Rows

Rows can be deleted from a table in three distinct manners, each mutually exclusive.

  • By specifying row IDs
  • By applying a filter
  • By removing all rows
const { deletedRows } = await client.deleteTableRows({
  table: 'Data1Table',
  // Specify a list of IDs
  ids: [1, 2, 3],
  // Or use a filter (caution: all rows matching the filter will be deleted. It's advisable to first test with the findTableRows query)
  filter: { name: 'John Doe' },
  // Deletes every row within this table, irreversible action
  deleteAllRows: true
})

Upsert Table Rows

The upsert operation is a combination of insert and update, allowing you to insert new rows or update existing ones. The operation is based on the row's ID.

keyColumn: The column used to identify the row. This column must be unique. Failure occurs if more than one row is found for the same key.

const { rows, inserted, updated, errors, warnings } = await client.upsertTableRows({
  table: 'Data1Table',
  keyColumn: 'id',
  rows: [
    {
      name: 'test'
    }
  ]
})

Create Table

To create a table, simply provide a name and a schema. The naming convention follows the same rules as those applied within the studio. Initially, the schema can be empty and populated or modified later. The schema accepts either an example object, from which it will infer the structure, or a directly provided JSON schema.

const { table } = await client.createTable({
  name: 'Data1Table',
  schema: {
    name: 'test'
  }
})

Example using a JSON schema. A field marked as 'searchable' will be indexed, enhancing its availability for search queries. Note, 'index' is a system-managed, read-only attribute.

const { table } = await client.createTable({
  name: 'Data1Table',
  schema: {
    type: 'object',
    'x-zui': {},
    properties: {
      name: {
        type: 'string',
        'x-zui': {
          index: 0,
          searchable: true
        },
        nullable: true
      }
    },
    additionalProperties: true
  }
})

Update Table

You can update a table's name, schema, or both. Providing only one of these attributes is also permissible. Should you alter the type of an existing column, be advised that all its entries will default to NULL.

Adding a new field

const { table } = await client.updateTable({
  table: 'Data1Table',
  schema: {
    myNewField: 'abc', // Direct value assignment
    // Or specify with a JSON schema
    myNewField: { type: 'string' }
  }
})

Removing an existing field:

const { table } = await client.updateTable({
  table: 'Data1Table',
  schema: {
    myOldField: null // Field removal
  }
})

Renaming the table:

await client.updateTable({ table: 'Data1Table', name: 'myNewTable' })

Rename Table Column:

To rename a table column, specify both the current and new names. The column's data type and attributes will remain unchanged.

const { table } = await client.renameTableColumn({
  table: 'Data1Table',
  name: 'My Old Column Name',
  newName: 'New Column Name'
})

Get Table

To retrieve a table's schema, use the getTable function. This will return the table's name and schema.

const { table } = await client.getTable({ table: 'Data1Table' })

Delete Table

To delete a table, use the deleteTable function. This will permanently remove the table and all its records.

const { table } = await client.deleteTable({ table: 'Data1Table' })

Operations on Tables - via Code

Studio Explorer

createRecord

To create a new record in a table, use the createRecord function. You can specify the values for each field in the record. This function can be particularly useful when you need to store new information, such as a new user's details or a newly completed transaction.

// Create a record in the table
await Data1Table.createRecord({ Age: 25, Name: 'Tom', Occupation: 'Student' })

createRecords

To create multiple records in a table, use the createRecords function. You can specify the values for each field in the record. Useful for creating multiple records at once.

// Creating two records in the table
await Data1Table.createRecords([
  { Name: 'Tom', Age: 25, Occupation: 'Student', 'Is Friendly': true },
  { Name: 'Didier', Age: 54, Occupation: 'Pilot', 'Is Friendly': true },
])

deleteRecord

To delete a record in a table, use the deleteRecord function. You need to specify the record ID of the record you want to delete.

// Delete a record in the table at record ID 7 
await Data1Table.deleteRecord(7) 

deleteRecords

To delete multiple records in a table, use the deleteRecords function. You need to specify the record IDs of the records you want to delete in an array.

 // Delete records in the table at record IDs 7, 8, 9 
await Data1Table.deleteRecords([7, 8, 9]) 

findRecords

To find records in a table, use the findRecords function. You can add filters, just query everything or search using semantic search (like a Google Search!).

Basic usage and pagination

To retrieve data from a table in Botpress, you can use the findRecords function. This function allows you to specify the limit (number of records to return) and the offset (number of rows to start from).

For instance, if you want the first 10 records, you can set the limit to 10 and the offset to 0. To fetch the next 10 records, you can set the limit to 10 and the offset to 10.

To determine if there are more records available, you can check the length of the array returned. If the length is greater than 0, it indicates the presence of additional records.

Here's an example:

const limit = 10
const startRow = 0
 
workflow.records = await DataTable1.findRecords({
  limit,
  offset: startRow,
})
 
// Checking the second page
const secondPage = await Data1Table.findRecords({
  limit,
  filter: AI`everything`,
  offset: startRow + limit,
})
 
workflow.hasNextPage = secondPage.length > 0 // This flag can be consumed in your bot flow to handle pagination

You can apply a similar approach to determine if there are previous pages as well.

Searching fields with natural language

To perform a traditional search, use the search parameter when calling the DataTable1.findRecords() function in Botpress. Here is an example:

workflow.records = await DataTable1.findRecords({ search: 'red velvet chairs' })

You can pass variables like event.preview to the search parameter to search for user input or any other variable. This allows you to preprocess the user's input before performing the search.

Filtering results

You can filter results using filters. Similar to MongoDB, this code provides methods to filter results. The filtering mechanism can be divided into two main categories: Primitive and Logical.

Don't want to deal with code? Skip to the next section for AI generated filters.

Primitive Filter

Primitive filters are easy and simple to use, allowing you to match the exact values you are looking for. Here are some of the primitives you can use:

  • $eq: Matches values that are equal to a specified value.

    { "name": { "$eq": "John" } }
  • $gt: Matches values that are greater than a specified value.

    { "age": { "$gt": 20 } }
  • $gte: Matches values that are greater than or equal to a specified value.

    { "age": { "$gte": 20 } }
  • $lt: Matches values that are less than a specified value.

    { "age": { "$lt": 20 } }
  • $lte: Matches values that are less than or equal to a specified value.

    { "age": { "$lte": 20 } }
  • $ne: Matches all values that are not equal to a specified value.

    { "name": { "$ne": "John" } }
  • $in: Matches any of the values specified in an array.

    { "name": { "$in": ["Alice", "Bob", "John"] } }
  • $nin: Matches none of the values specified in an array.

    { "name": { "$nin": ["Alice", "Bob", "John"] } }
  • $exists: Matches documents that have the specified field.

    { "name": { "$exists": true } }
  • $mod: Performs modulo operation on the value of field and matches documents where field % divisor equals the specified remainder.

    { "qty": { "$mod": [4, 0] } }
  • $size: Matches any document where an array field contains a specified number of elements.

    { "tags": { "$size": 3 } }
  • $regex: Provides regular expression capabilities for pattern matching strings in queries. It uses a placeholder to prevent SQL injections.

    { "name": { "$regex": "[a-z]" } }
  • $options: Modifies the $regex operator to enable options. Currently, there are two options, 'i' for case insensitive and 'c' for case sensitive.

    { "name": { "$regex": "[a-z]", "$options": 'i' } } // case insensitive
    
    { "name": { "$regex": "[a-z]", "$options": 'c' } } // case sensitive

Logical Filter

Logical operators allow you to connect more filtering clauses.

  • $and: Joins query clauses with a logical AND. Returns all documents that match the conditions of both clauses.

    { "$and": [{ "price": { "$ne": 1.99 } }, { "price": { "$exists": true } }] }
  • $or: Joins query clauses with a logical OR. Returns all documents that match the conditions of either clause.

    { "$or": [{ "price": { "$eq": 1.99 } }, { "price": { "$exists": false } }] }
  • $not: Inverts the effect of a query expression and returns documents that do not match the query expression.

    { "$not": { "price": { "$eq": 1.99 } } }

Please note that these filters and operators can be used together to create complex queries to exactly fit your needs. You can also nest filters like so :

{
   "$and": [
      { "price": { "$lte": 19.99 }},
      { "$or":  [
         {"category": { "$eq": "Shoes" }},
         {"category": { "$eq": "Hats"}}
        ]
      }
   ]                          
 }

Using AI to generate filters

Thankfully, Botpress comes with artificial intelligence (AI) capabilities when it comes to filtering records. This means you can provide a human-like query, and the AI will intelligently decipher it and create the right filters for your query.

This can significantly simplify the construction of complex queries and reduce the likelihood of human errors in the process.

const leads = await LeadsTable.findRecords({
  filter: AI`works either at ${workflow.company} OR is not in IT`,
  limit: 10,
  offset: 0,
})

The query is written inside the backticks (`) that come after `filter: AI`. The AI expects a query string, and that's why backticks are used.

Re-using filters (selectors)

Selectors are filters that you can re-use. They don't have to be written again and again.

For example, you might use the query selector "People with age greater than 18" to find all users who are above 18.

  1. Create a query selector in the table and give it a name. (e.g. adults)
  2. Add query as People with age greater than 18
  3. use below code to find records and save it in a variable called adults
// Find records in the table
const data = await Data1Table.findRecords({ selectorName: 'adults' })
workflow.adults = data

Difference between search and AI filters

A standard Search in Botpress is employed to identify specific strings like a user's name. It gives precision-based results by matching specific sets of characters.

On the other hand, an AI Filter broadens the scope and is used primarily for filtering results rather than explicitly searching them. It helps you track down users based on certain criteria, such as age above 25.

Although the AI Filter is not primarily designed for exact string hunting, it's flexible enough to allow it. For instance, you could use it to filter results for Canada as a country.

getRecord

You can use the getRecord function to retrieve a specific record from a table based on a unique identifier. For example, you might use the getRecord function to retrieve a user's profile information based on their unique user ID.

 
// 1 is the record ID and User is the variable name
workflow.User = data
 

updateRecord

You can use the updateRecord function to modify a specific record in a table. This will be useful when you need to update a specific field in the record. For example, you might use this function to update a user's profile information or to update the status of a transaction.

// Update at record ID 8's Age to 45
await Data1Table.updateRecord(8, { Age: 45 })

upsertRecords

The upsertRecords function allows you to update or create records in a table in a single operation. It is useful for any situation where you need to update or insert data into a table without having to worry about whether the rows already exist.

Example:

// Update the name and address of the records with IDs 1 and 2.
await CustomersTable.upsertRecords('id', [
  { id: 1, name: 'John Doe', address: '123 Main Street' },
  { id: 2, name: 'Jane Doe', address: '456 Elm Street' },
])

This code will upsert the records into the CustomersTable. If the records with IDs 1 and 2 already exist in the table, the function will update the existing rows with the new name and address values. If the records with IDs 1 and 2 don't already exist in the table, the function will insert new rows into the table with the new name and address values.

Importing and Exporting

Import a Table

You can import a table from a CSV file. Click on the Import Table button above the table and select the CSV file you want to import. Botpress will automatically detect the column names and types from the CSV file and create the table structure for you.

Export a Table

Click on the Export Table button to export the table. This will download the table as a CSV file to your computer.

đź’ˇ

Don't edit the CSV file directly to avoid data loss and import errors. Instead, make the changes in Botpress Studio and then export the updated table again.

When you export a bot the tables are exported as well so there is no need to export them separately unless you want to use them in another bot.