RBQL is an eval-based SQL-like query engine for (not only) CSV file processing. It provides SQL-like language that supports SELECT queries with Python expressions.
RBQL is best suited for data transformation, data cleaning, and analytical queries.
[Official Site](https://rbql.org/)
### Using RBQL as a Python library
If you want to use rbql module as a python library for your own app - please read [API Documentation](https://github.com/mechatroner/RBQL/blob/master/rbql-py/LIBRARY_README.md#using-rbql-as-python-library)
The rbql module includes functions to query csv tables, pandas dataframes, python lists and more.
### Using RBQL as IPython "magic" command
#### Syntax
`%rbql <query>`
#### Usage example
```
from vega_datasets import data
my_cars = data.cars()
%load_ext rbql
result_df = %rbql SELECT a.Name, a.Weight_in_lbs / 1000, a.Horsepower FROM my_cars WHERE a.Horsepower > 100 ORDER BY a.Weight_in_lbs DESC LIMIT 15
result_df
```
You can also visit the [demo Google Colab notebook](https://colab.research.google.com/drive/1_cFPtnQUxILP0RE2_DBlqIfXaEzT-oZ6?usp=sharing) to try the `%rbql` "magic" command.
### Using RBQL as a CLI App
#### Installation:
```
$ pip install rbql
```
#### Usage example (non-interactive mode):
```
$ rbql-py --query "select a1, a2 order by a1" --delim , < input.csv
```
#### Usage example (interactive mode):
In interactive mode rbql-py will show input table preview so it is easier to type SQL-like query.
```
$ rbql-py --input input.csv --output result.csv
```
### Main Features
* Use Python expressions inside _SELECT_, _UPDATE_, _WHERE_ and _ORDER BY_ statements
* Supports multiple input formats
* Result set of any query immediately becomes a first-class table on its own
* No need to provide FROM statement in the query - input table is defined by the current context
* Supports all main SQL keywords
* Supports aggregate functions and GROUP BY queries
* Supports user-defined functions (UDF)
* Provides some new useful query modes which traditional SQL engines do not have
* Lightweight, dependency-free, works out of the box
#### Limitations:
* RBQL doesn't support nested queries, but they can be emulated with consecutive queries
* Number of tables in all JOIN queries is always 2 (input table and join table), use consecutive queries to join 3 or more tables
### Supported SQL Keywords (Keywords are case insensitive)
* SELECT
* UPDATE
* WHERE
* ORDER BY ... [ DESC | ASC ]
* [ LEFT | INNER ] JOIN
* DISTINCT
* GROUP BY
* TOP _N_
* LIMIT _N_
* AS
All keywords have the same meaning as in SQL queries. You can check them [online](https://www.w3schools.com/sql/default.asp)
### RBQL variables
RBQL for CSV files provides the following variables which you can use in your queries:
* _a1_, _a2_,..., _a{N}_
Variable type: **string**
Description: value of i-th field in the current record in input table
* _b1_, _b2_,..., _b{N}_
Variable type: **string**
Description: value of i-th field in the current record in join table B
* _NR_
Variable type: **integer**
Description: Record number (1-based)
* _NF_
Variable type: **integer**
Description: Number of fields in the current record
* _a.name_, _b.Person_age_, ... _a.{Good_alphanumeric_column_name}_
Variable type: **string**
Description: Value of the field referenced by it's "name". You can use this notation if the field in the header has a "good" alphanumeric name
* _a["object id"]_, _a['9.12341234']_, _b["%$ !! 10 20"]_ ... _a["Arbitrary column name!"]_
Variable type: **string**
Description: Value of the field referenced by it's "name". You can use this notation to reference fields by arbitrary values in the header
### UPDATE statement
_UPDATE_ query produces a new table where original values are replaced according to the UPDATE expression, so it can also be considered a special type of SELECT query.
### Aggregate functions and queries
RBQL supports the following aggregate functions, which can also be used with _GROUP BY_ keyword:
_COUNT_, _ARRAY_AGG_, _MIN_, _MAX_, _SUM_, _AVG_, _VARIANCE_, _MEDIAN_
Limitation: aggregate functions inside Python expressions are not supported. Although you can use expressions inside aggregate functions.
E.g. `MAX(float(a1) / 1000)` - valid; `MAX(a1) / 1000` - invalid.
There is a workaround for the limitation above for _ARRAY_AGG_ function which supports an optional parameter - a callback function that can do something with the aggregated array. Example:
`SELECT a2, ARRAY_AGG(a1, lambda v: sorted(v)[:5]) GROUP BY a2` - Python; `SELECT a2, ARRAY_AGG(a1, v => v.sort().slice(0, 5)) GROUP BY a2` - JS
### JOIN statements
Join table B can be referenced either by its file path or by its name - an arbitrary string which the user should provide before executing the JOIN query.
RBQL supports _STRICT LEFT JOIN_ which is like _LEFT JOIN_, but generates an error if any key in the left table "A" doesn't have exactly one matching key in the right table "B".
Table B path can be either relative to the working dir, relative to the main table or absolute.
Limitation: _JOIN_ statements can't contain Python/JS expressions and must have the following form: _<JOIN\_KEYWORD> (/path/to/table.tsv | table_name ) ON a... == b... [AND a... == b... [AND ... ]]_
### SELECT EXCEPT statement
SELECT EXCEPT can be used to select everything except specific columns. E.g. to select everything but columns 2 and 4, run: `SELECT * EXCEPT a2, a4`
Traditional SQL engines do not support this query mode.
### UNNEST() operator
UNNEST(list) takes a list/array as an argument and repeats the output record multiple times - one time for each value from the list argument.
Example: `SELECT a1, UNNEST(a2.split(';'))`
### LIKE() function
RBQL does not support LIKE operator, instead it provides "like()" function which can be used like this:
`SELECT * where like(a1, 'foo%bar')`
### WITH (header) and WITH (noheader) statements
You can set whether the input (and join) CSV file has a header or not using the environment configuration parameters which could be `--with_headers` CLI flag or GUI checkbox or something else.
But it is also possible to override this selection directly in the query by adding either `WITH (header)` or `WITH (noheader)` statement at the end of the query.
Example: `select top 5 NR, * with (header)`
### User Defined Functions (UDF)
RBQL supports User Defined Functions
You can define custom functions and/or import libraries in a special file:
* `~/.rbql_init_source.py` - for Python
## Examples of RBQL queries
#### With Python expressions
* `SELECT TOP 100 a1, int(a2) * 10, len(a4) WHERE a1 == "Buy" ORDER BY int(a2) DESC`
* `SELECT a.id, a.weight / 1000 AS weight_kg`
* `SELECT * ORDER BY random.random()` - random sort
* `SELECT len(a.vehicle_price) / 10, a2 WHERE int(a.vehicle_price) < 500 and a['Vehicle type'] in ["car", "plane", "boat"] limit 20` - referencing columns by names from header and using Python's "in" to emulate SQL's "in"
* `UPDATE SET a3 = 'NPC' WHERE a3.find('Non-playable character') != -1`
* `SELECT NR, *` - enumerate records, NR is 1-based
* `SELECT * WHERE re.match(".*ab.*", a1) is not None` - select entries where first column has "ab" pattern
* `SELECT a1, b1, b2 INNER JOIN ./countries.txt ON a2 == b1 ORDER BY a1, a3` - example of join query
* `SELECT MAX(a1), MIN(a1) WHERE a.Name != 'John' GROUP BY a2, a3` - example of aggregate query
* `SELECT *a1.split(':')` - Using Python3 unpack operator to split one column into many. Do not try this with other SQL engines!
### RBQL design principles and architecture
RBQL core idea is based on dynamic code generation and execution with [exec](https://docs.python.org/3/library/functions.html#exec) and [eval](https://www.w3schools.com/jsref/jsref_eval.asp) functions.
Here are the main steps that RBQL engine performs when processing a query:
1. Shallow parsing: split the query into logical expressions such as "SELECT", "WHERE", "ORDER BY", etc.
2. Embed the expression segments into the main loop template code
3. Execute the hydrated loop code
Here you can find a very basic working script (only 15 lines of Python code) which implements this idea: [mini_rbql.py](https://github.com/mechatroner/mini-rbql/blob/master/mini_rbql.py)
The diagram below gives an overview of the main RBQL components and data flow:
![RBQL Diagram](https://i.imgur.com/KDQHoVM.png)
### Advantages of RBQL over traditional SQL engines
* Provides power and flexibility of general purpose Python and JS languages in relational expressions (including regexp, math, file system, json, xml, random and many other libraries that these languages provide)
* Can work with different data sources including CSV files, sqlite tables, native 2D arrays/lists (traditional SQL engines are usually tightly coupled with their databases)
* Result set of any query immediately becomes a first-class table on its own
* Supports both TOP and LIMIT keywords
* Provides additional NR (record number) variable which is especially useful for input sources where record order is well defined (such as CSV files)
* Supports input tables with inconsistent number of fields per record
* Allows to generate result sets with variable number of fields per record e.g. by using split() function and unpack operator (Python) / destructuring assignment (JS)
* UPDATE is a special case of SELECT query - this prevents accidental data loss
* No need to use FROM statement - the table name is defined by the context. This improves query typing speed and allows immediate autocomplete for variables inside SELECT statement (in traditional SQL engines autocomplete will not work until you write FROM statement, which goes after SELECT statement)
* SELECT, WHERE, ORDER BY, and other statements can be rearranged in any way you like
* Supports EXCEPT statement
* Provides a fully-functional client-side browser demo application
* Almost nonexistent entry barrier both for SQL users and JS/Python users
* Integration with popular text editors (VSCode, Vim, Sublime Text, Atom)
* Small, maintainable, dependency-free, eco-friendly and hackable code base: RBQL engine fits into a single file with less than 2000 LOC
### Disadvantages of RBQL compared to traditional SQL engines
* Not suitable for transactional workload
* RBQL doesn't support nested queries, but they can be emulated with consecutive queries
* Number of tables in all JOIN queries is always 2 (input table and join table), use consecutive queries to join 3 or more tables
* Does not support HAVING statement
### References
* [RBQL: Official Site](https://rbql.org/)
* rbql-js library and CLI App for Node.js - [npm](https://www.npmjs.com/package/rbql)
* Rainbow CSV extension with integrated RBQL in [Visual Studio Code](https://marketplace.visualstudio.com/items?itemName=mechatroner.rainbow-csv)
* Rainbow CSV extension with integrated RBQL in [Vim](https://github.com/mechatroner/rainbow_csv)
* Rainbow CSV extension with integrated RBQL in [Sublime Text 3](https://packagecontrol.io/packages/rainbow_csv)
* Rainbow CSV extension with integrated RBQL in [Atom](https://atom.io/packages/rainbow-csv)
Raw data
{
"_id": null,
"home_page": "https://github.com/mechatroner/RBQL",
"name": "rbql",
"maintainer": "",
"docs_url": null,
"requires_python": "",
"maintainer_email": "",
"keywords": "SQL SQL-like transpiler RBQL library CLI command-line CSV TSV IPython Jupyter",
"author": "Dmitry Ignatovich",
"author_email": "mechatroner@yandex.ru",
"download_url": "https://files.pythonhosted.org/packages/c0/09/124d971d6c97e901ebea76676626309b3270773327e6945966ffb67754b5/rbql-0.27.0.tar.gz",
"platform": null,
"description": "RBQL is an eval-based SQL-like query engine for (not only) CSV file processing. It provides SQL-like language that supports SELECT queries with Python expressions. \nRBQL is best suited for data transformation, data cleaning, and analytical queries. \n\n[Official Site](https://rbql.org/)\n\n\n### Using RBQL as a Python library\n\nIf you want to use rbql module as a python library for your own app - please read [API Documentation](https://github.com/mechatroner/RBQL/blob/master/rbql-py/LIBRARY_README.md#using-rbql-as-python-library) \nThe rbql module includes functions to query csv tables, pandas dataframes, python lists and more.\n\n### Using RBQL as IPython \"magic\" command\n\n#### Syntax\n`%rbql <query>`\n\n#### Usage example\n```\nfrom vega_datasets import data\nmy_cars = data.cars()\n%load_ext rbql\nresult_df = %rbql SELECT a.Name, a.Weight_in_lbs / 1000, a.Horsepower FROM my_cars WHERE a.Horsepower > 100 ORDER BY a.Weight_in_lbs DESC LIMIT 15\nresult_df\n```\nYou can also visit the [demo Google Colab notebook](https://colab.research.google.com/drive/1_cFPtnQUxILP0RE2_DBlqIfXaEzT-oZ6?usp=sharing) to try the `%rbql` \"magic\" command.\n\n### Using RBQL as a CLI App\n\n#### Installation:\n```\n$ pip install rbql\n```\n\n#### Usage example (non-interactive mode):\n```\n$ rbql-py --query \"select a1, a2 order by a1\" --delim , < input.csv\n```\n\n#### Usage example (interactive mode):\nIn interactive mode rbql-py will show input table preview so it is easier to type SQL-like query. \n```\n$ rbql-py --input input.csv --output result.csv\n```\n\n\n### Main Features\n\n* Use Python expressions inside _SELECT_, _UPDATE_, _WHERE_ and _ORDER BY_ statements\n* Supports multiple input formats\n* Result set of any query immediately becomes a first-class table on its own\n* No need to provide FROM statement in the query - input table is defined by the current context\n* Supports all main SQL keywords\n* Supports aggregate functions and GROUP BY queries\n* Supports user-defined functions (UDF)\n* Provides some new useful query modes which traditional SQL engines do not have\n* Lightweight, dependency-free, works out of the box\n\n#### Limitations:\n\n* RBQL doesn't support nested queries, but they can be emulated with consecutive queries\n* Number of tables in all JOIN queries is always 2 (input table and join table), use consecutive queries to join 3 or more tables\n\n### Supported SQL Keywords (Keywords are case insensitive)\n\n* SELECT\n* UPDATE\n* WHERE\n* ORDER BY ... [ DESC | ASC ]\n* [ LEFT | INNER ] JOIN\n* DISTINCT\n* GROUP BY\n* TOP _N_\n* LIMIT _N_\n* AS\n\nAll keywords have the same meaning as in SQL queries. You can check them [online](https://www.w3schools.com/sql/default.asp) \n\n\n### RBQL variables\nRBQL for CSV files provides the following variables which you can use in your queries:\n\n* _a1_, _a2_,..., _a{N}_ \n Variable type: **string** \n Description: value of i-th field in the current record in input table \n* _b1_, _b2_,..., _b{N}_ \n Variable type: **string** \n Description: value of i-th field in the current record in join table B \n* _NR_ \n Variable type: **integer** \n Description: Record number (1-based) \n* _NF_ \n Variable type: **integer** \n Description: Number of fields in the current record \n* _a.name_, _b.Person_age_, ... _a.{Good_alphanumeric_column_name}_ \n Variable type: **string** \n Description: Value of the field referenced by it's \"name\". You can use this notation if the field in the header has a \"good\" alphanumeric name \n* _a[\"object id\"]_, _a['9.12341234']_, _b[\"%$ !! 10 20\"]_ ... _a[\"Arbitrary column name!\"]_ \n Variable type: **string** \n Description: Value of the field referenced by it's \"name\". You can use this notation to reference fields by arbitrary values in the header\n\n\n### UPDATE statement\n\n_UPDATE_ query produces a new table where original values are replaced according to the UPDATE expression, so it can also be considered a special type of SELECT query.\n\n### Aggregate functions and queries\n\nRBQL supports the following aggregate functions, which can also be used with _GROUP BY_ keyword: \n_COUNT_, _ARRAY_AGG_, _MIN_, _MAX_, _SUM_, _AVG_, _VARIANCE_, _MEDIAN_ \n\nLimitation: aggregate functions inside Python expressions are not supported. Although you can use expressions inside aggregate functions. \nE.g. `MAX(float(a1) / 1000)` - valid; `MAX(a1) / 1000` - invalid. \nThere is a workaround for the limitation above for _ARRAY_AGG_ function which supports an optional parameter - a callback function that can do something with the aggregated array. Example: \n`SELECT a2, ARRAY_AGG(a1, lambda v: sorted(v)[:5]) GROUP BY a2` - Python; `SELECT a2, ARRAY_AGG(a1, v => v.sort().slice(0, 5)) GROUP BY a2` - JS\n\n\n### JOIN statements\n\nJoin table B can be referenced either by its file path or by its name - an arbitrary string which the user should provide before executing the JOIN query. \nRBQL supports _STRICT LEFT JOIN_ which is like _LEFT JOIN_, but generates an error if any key in the left table \"A\" doesn't have exactly one matching key in the right table \"B\". \nTable B path can be either relative to the working dir, relative to the main table or absolute. \nLimitation: _JOIN_ statements can't contain Python/JS expressions and must have the following form: _<JOIN\\_KEYWORD> (/path/to/table.tsv | table_name ) ON a... == b... [AND a... == b... [AND ... ]]_\n\n### SELECT EXCEPT statement\n\nSELECT EXCEPT can be used to select everything except specific columns. E.g. to select everything but columns 2 and 4, run: `SELECT * EXCEPT a2, a4` \nTraditional SQL engines do not support this query mode.\n\n\n### UNNEST() operator\nUNNEST(list) takes a list/array as an argument and repeats the output record multiple times - one time for each value from the list argument. \nExample: `SELECT a1, UNNEST(a2.split(';'))` \n\n\n### LIKE() function\nRBQL does not support LIKE operator, instead it provides \"like()\" function which can be used like this:\n`SELECT * where like(a1, 'foo%bar')`\n\n\n### WITH (header) and WITH (noheader) statements\nYou can set whether the input (and join) CSV file has a header or not using the environment configuration parameters which could be `--with_headers` CLI flag or GUI checkbox or something else.\nBut it is also possible to override this selection directly in the query by adding either `WITH (header)` or `WITH (noheader)` statement at the end of the query.\nExample: `select top 5 NR, * with (header)`\n\n\n### User Defined Functions (UDF)\n\nRBQL supports User Defined Functions \nYou can define custom functions and/or import libraries in a special file: \n* `~/.rbql_init_source.py` - for Python\n\n\n## Examples of RBQL queries\n\n#### With Python expressions\n\n* `SELECT TOP 100 a1, int(a2) * 10, len(a4) WHERE a1 == \"Buy\" ORDER BY int(a2) DESC`\n* `SELECT a.id, a.weight / 1000 AS weight_kg`\n* `SELECT * ORDER BY random.random()` - random sort\n* `SELECT len(a.vehicle_price) / 10, a2 WHERE int(a.vehicle_price) < 500 and a['Vehicle type'] in [\"car\", \"plane\", \"boat\"] limit 20` - referencing columns by names from header and using Python's \"in\" to emulate SQL's \"in\"\n* `UPDATE SET a3 = 'NPC' WHERE a3.find('Non-playable character') != -1`\n* `SELECT NR, *` - enumerate records, NR is 1-based\n* `SELECT * WHERE re.match(\".*ab.*\", a1) is not None` - select entries where first column has \"ab\" pattern\n* `SELECT a1, b1, b2 INNER JOIN ./countries.txt ON a2 == b1 ORDER BY a1, a3` - example of join query\n* `SELECT MAX(a1), MIN(a1) WHERE a.Name != 'John' GROUP BY a2, a3` - example of aggregate query\n* `SELECT *a1.split(':')` - Using Python3 unpack operator to split one column into many. Do not try this with other SQL engines!\n\n\n### RBQL design principles and architecture\nRBQL core idea is based on dynamic code generation and execution with [exec](https://docs.python.org/3/library/functions.html#exec) and [eval](https://www.w3schools.com/jsref/jsref_eval.asp) functions.\nHere are the main steps that RBQL engine performs when processing a query:\n1. Shallow parsing: split the query into logical expressions such as \"SELECT\", \"WHERE\", \"ORDER BY\", etc.\n2. Embed the expression segments into the main loop template code\n3. Execute the hydrated loop code\n\nHere you can find a very basic working script (only 15 lines of Python code) which implements this idea: [mini_rbql.py](https://github.com/mechatroner/mini-rbql/blob/master/mini_rbql.py)\n\nThe diagram below gives an overview of the main RBQL components and data flow:\n![RBQL Diagram](https://i.imgur.com/KDQHoVM.png)\n\n\n### Advantages of RBQL over traditional SQL engines\n* Provides power and flexibility of general purpose Python and JS languages in relational expressions (including regexp, math, file system, json, xml, random and many other libraries that these languages provide)\n* Can work with different data sources including CSV files, sqlite tables, native 2D arrays/lists (traditional SQL engines are usually tightly coupled with their databases)\n* Result set of any query immediately becomes a first-class table on its own\n* Supports both TOP and LIMIT keywords\n* Provides additional NR (record number) variable which is especially useful for input sources where record order is well defined (such as CSV files)\n* Supports input tables with inconsistent number of fields per record\n* Allows to generate result sets with variable number of fields per record e.g. by using split() function and unpack operator (Python) / destructuring assignment (JS)\n* UPDATE is a special case of SELECT query - this prevents accidental data loss\n* No need to use FROM statement - the table name is defined by the context. This improves query typing speed and allows immediate autocomplete for variables inside SELECT statement (in traditional SQL engines autocomplete will not work until you write FROM statement, which goes after SELECT statement)\n* SELECT, WHERE, ORDER BY, and other statements can be rearranged in any way you like\n* Supports EXCEPT statement\n* Provides a fully-functional client-side browser demo application\n* Almost nonexistent entry barrier both for SQL users and JS/Python users\n* Integration with popular text editors (VSCode, Vim, Sublime Text, Atom)\n* Small, maintainable, dependency-free, eco-friendly and hackable code base: RBQL engine fits into a single file with less than 2000 LOC\n\n### Disadvantages of RBQL compared to traditional SQL engines\n* Not suitable for transactional workload\n* RBQL doesn't support nested queries, but they can be emulated with consecutive queries\n* Number of tables in all JOIN queries is always 2 (input table and join table), use consecutive queries to join 3 or more tables\n* Does not support HAVING statement\n\n\n### References\n\n* [RBQL: Official Site](https://rbql.org/)\n* rbql-js library and CLI App for Node.js - [npm](https://www.npmjs.com/package/rbql) \n* Rainbow CSV extension with integrated RBQL in [Visual Studio Code](https://marketplace.visualstudio.com/items?itemName=mechatroner.rainbow-csv) \n* Rainbow CSV extension with integrated RBQL in [Vim](https://github.com/mechatroner/rainbow_csv) \n* Rainbow CSV extension with integrated RBQL in [Sublime Text 3](https://packagecontrol.io/packages/rainbow_csv) \n* Rainbow CSV extension with integrated RBQL in [Atom](https://atom.io/packages/rainbow-csv)\n\n\n",
"bugtrack_url": null,
"license": "",
"summary": "Rainbow Query Language",
"version": "0.27.0",
"split_keywords": [
"sql",
"sql-like",
"transpiler",
"rbql",
"library",
"cli",
"command-line",
"csv",
"tsv",
"ipython",
"jupyter"
],
"urls": [
{
"comment_text": "",
"digests": {
"md5": "2b50d6eb99ea8aefd3de4d6f8d002766",
"sha256": "020b9c277fecc515283421278fb46ec0745d73d2195bf01d3bb8c9d8fc0fc03d"
},
"downloads": -1,
"filename": "rbql-0.27.0-py3-none-any.whl",
"has_sig": false,
"md5_digest": "2b50d6eb99ea8aefd3de4d6f8d002766",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": null,
"size": 41228,
"upload_time": "2022-12-03T00:42:21",
"upload_time_iso_8601": "2022-12-03T00:42:21.522440Z",
"url": "https://files.pythonhosted.org/packages/b5/b1/587ca52e28a2ca8f97d140b55195fb7527cc774e83cb9464eec0fd08c29d/rbql-0.27.0-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"md5": "3848e171e3e3ad95aa2a506c71706f3e",
"sha256": "c5fe269b0fd0aba8aeee4d55ac15cb743ba9122045c507ea0fd12f493ccae481"
},
"downloads": -1,
"filename": "rbql-0.27.0.tar.gz",
"has_sig": false,
"md5_digest": "3848e171e3e3ad95aa2a506c71706f3e",
"packagetype": "sdist",
"python_version": "source",
"requires_python": null,
"size": 41337,
"upload_time": "2022-12-03T00:42:23",
"upload_time_iso_8601": "2022-12-03T00:42:23.248704Z",
"url": "https://files.pythonhosted.org/packages/c0/09/124d971d6c97e901ebea76676626309b3270773327e6945966ffb67754b5/rbql-0.27.0.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2022-12-03 00:42:23",
"github": true,
"gitlab": false,
"bitbucket": false,
"github_user": "mechatroner",
"github_project": "RBQL",
"travis_ci": false,
"coveralls": false,
"github_actions": false,
"lcname": "rbql"
}