2022-10-28 14:17:40 +00:00
## Query
The `#query` is the most widely used directive. It can be used to query various data sources and render results in various ways.
### Syntax
1. _start with_ : `<!-- #query [QUERY GOES HERE] -->`
2. _end with_ : `<!-- /query -->`
3. _write your query_ : replace `[QUERY GOES HERE]` with any query you want using the options below.
2023-07-29 13:02:37 +00:00
4. _available query options_ : Usage of options is similar to SQL except for the special `render` option. The `render` option is used to display the data in a format that you created in a separate template.
2022-10-28 14:17:40 +00:00
* `where`
* `order by`
* `limit`
* `select`
* `render`
P.S.: If you are a developer or have a technical knowledge to read a code and would like to know more about syntax, please check out
2023-01-04 18:50:36 +00:00
[query grammar ](https://github.com/silverbulletmd/silverbullet/blob/main/common/markdown_parser/query.grammar ).
2022-10-28 14:17:40 +00:00
#### 2.1. Available query operators:
- `=` equals
- `!=` not equals
- `<` less than
- `<=` less than or equals
- `>` greater than
- `>=` greater than or equals
- `=~` to match against a regular expression
- `!=~` does not match this regular expression
- `in` member of a list (e.g. `prop in ["foo", "bar"]` )
Further, you can combine multiple of these with `and` . Example
`prop =~ /something/ and prop != “something”` .
### 3. How to run a query?
After writing the query, there are three options:
1. Open the **command palette** and run {[Directives: Update]}
2. Use shortcut: hit **Alt-q** (Windows, Linux) or **Option-q** (Mac)
3. Go to another page and come back to the page where the query is located, it always updates when a page is loaded
After using one of the options, the “body” of the query is replaced with the new results of the query data will be displayed.
### 4. Data sources
Available data sources can be categorized as:
1. Builtin data sources
2. Data that can be inserted by users
3. Plug’ s data sources
The best part about data sources: there is auto-completion. 🎉
Start writing `<!— #query` or simply use `/query` slash command, it will show you all available data sources. 🤯
2023-07-24 09:09:17 +00:00
Additionally there are [[🔌 Core/Templates@vars|special variables]] you can use in your queries.
2023-01-04 18:54:24 +00:00
For example, if you wanted a query for all the tasks from a previous day's daily note, you could use the following query:
`<!-- #query task where page = "📅 {{yesterday}}" -->`
2022-10-28 14:17:40 +00:00
#### 4.1. Available data sources
- `page` : list of all pages
2022-12-28 07:51:55 +00:00
- `attachment` : list of all attachments
2023-01-04 18:17:55 +00:00
- `task` : list of all tasks (created with `[ ]` ) across all pages
2022-10-28 14:17:40 +00:00
- `full-text` : use it with `where phrase = "SOME_TEXT"` . List of all pages where `SOME_TEXT` is mentioned
2023-01-04 18:17:55 +00:00
- `item` : list of ordered and unordered items such as bulleted lists across all pages
2022-10-28 14:17:40 +00:00
- `tag` : list of all hashtags used in all pages
- `link` : list of all pages giving a link to the page where query is written
- `data` : You can insert data using the syntax below. You can query the data using the `data` source.
```data
name: John
age: 50
city: Milan
country: Italy
---
name: Jane
age: 53
city: Rome
country: Italy
---
name: Francesco
age: 28
city: Berlin
country: Germany
```
Example:
<!-- #query data where age > 20 and country = "Italy" -->
2023-05-23 18:53:53 +00:00
|name|age|city |country|page |pos |
|----|--|-----|-----|------------------|----|
2023-08-04 16:56:55 +00:00
|John|50|Milan|Italy|🔌 Directive/Query|2933|
|Jane|53|Rome |Italy|🔌 Directive/Query|2934|
2022-10-28 14:17:40 +00:00
<!-- /query -->
#### 4.2 Plugs’ data sources
Certain plugs can also provide special data sources to query specific data. Some examples are:
- [[🔌 Github]] provides `gh-pull` to query PRs for selected repo
- [[🔌 Mattermost]] provides `mm-saved` to fetch (by default 15) saved posts in
Mattermost
For a complete list of data sources, please check plugs’ own pages.
### 5. Templates
Templates are predefined formats to render the body of the query.
#### 5.1 How to create a template?
It is pretty easy. You just need to create a new page. However, it is
recommended to create your templates using `template/[TEMPLATE_NAME]`
2023-01-16 15:45:55 +00:00
convention. For this guide, we will create `template/plug` to display list of Plugs available in SilverBullet. We will use this template in the Examples section below.
2022-10-28 14:17:40 +00:00
#### 5.2 What is the syntax?
We are using Handlebars which is a simple templating language. It is using double curly braces and the name of the parameter to be injected. For our `template/plug` , we are using simple template like below.
* [[{{name}}]] by ** {{author}}** ([repo]({{repo}}))
Let me break it down for you
2023-01-16 15:45:55 +00:00
- `*` is creating a bullet point for each item in SilverBullet
2022-10-28 14:17:40 +00:00
- `[[{{name}}]]` is injecting the name of Plug and creating an internal link to
the page of the Plug
- `**{{author}}**` is injecting the author of the Plug and making it bold
- `([repo]({{repo}}))` is injecting the name of the Plug and creating an
external link to the GitHub page of the Plug
For more information on the Handlebars syntax, you can read the
[official documentation ](https://handlebarsjs.com/ ).
#### 5.3 How to use the template?
You just need to add the `render` keyword followed by the link of the template to the query like below:
2022-11-25 15:01:05 +00:00
<!-- #query page where type = "plug" render [[template/plug]] -->
<!-- /query -->
2022-10-28 14:17:40 +00:00
`#query page where type = "plug" render [[template/plug]]`
You can see the usage of our template in example 6.4 below.
### 6. Examples
We will walk you through a set of examples starting from a very basic one
through one formatting the data using templates.
2022-11-25 15:01:05 +00:00
Our goal in this exercise is to (i) get all plug pages (ii) ordered by last modified time and (iii) display in a nice format.
2022-10-28 14:17:40 +00:00
2022-11-25 15:01:05 +00:00
For the sake of simplicity, we will use the `page` data source and limit the results not to spoil the page.
2022-10-28 14:17:40 +00:00
#### 6.1 Simple query without any condition
**Goal:** We would like to get the list of all pages.
2022-11-25 15:01:05 +00:00
**Result:** Look at the data. This is more than we need. The query even gives us template pages. Let's try to limit it in the next step.
<!-- #query page limit 3 -->
2023-08-04 16:56:55 +00:00
|name |lastModified |contentType |size |perm|
|--|--|--|--|--|
|Authentication |1686682290943|text/markdown|1730 |rw|
|Guide/Deployment/Cloudflare and Portainer|1690298800145|text/markdown|12899|rw|
|Markdown |1676121406520|text/markdown|1178 |rw|
2022-10-28 14:17:40 +00:00
<!-- /query -->
2022-11-25 15:01:05 +00:00
2022-10-28 14:17:40 +00:00
#### 6.2 Simple query with a condition
**Goal:** We would like to get all plug pages sorted by last modified time.
2022-12-28 07:51:55 +00:00
**Result:** Okay, this is what we wanted but there is also information such as `perm` , `type` and `lastModified` that we don't need.
2022-10-28 14:17:40 +00:00
<!-- #query page where type = "plug" order by lastModified desc limit 5 -->
2023-08-04 16:56:55 +00:00
|name |lastModified |contentType |size|perm|type|uri |repo |author |share-support|
|--|--|--|--|--|--|--|--|--|--|
|🔌 Github |1691137925014|text/markdown|2206|rw|plug|github:silverbulletmd/silverbullet-github/github.plug.js |https://github.com/silverbulletmd/silverbullet-github |Zef Hemel|true|
|🔌 Mattermost|1691137924741|text/markdown|3535|rw|plug|github:silverbulletmd/silverbullet-mattermost/mattermost.plug.json|https://github.com/silverbulletmd/silverbullet-mattermost|Zef Hemel|true|
|🔌 Git |1691137924435|text/markdown|1112|rw|plug|github:silverbulletmd/silverbullet-git/git.plug.js |https://github.com/silverbulletmd/silverbullet-git |Zef Hemel| |
|🔌 Ghost |1691137922296|text/markdown|1733|rw|plug|github:silverbulletmd/silverbullet-ghost/ghost.plug.js |https://github.com/silverbulletmd/silverbullet-ghost |Zef Hemel|true|
|🔌 Share |1691137921643|text/markdown|693 |rw|plug| |https://github.com/silverbulletmd/silverbullet | | |
2022-10-28 14:17:40 +00:00
<!-- /query -->
#### 6.3 Query to select only certain fields
**Goal:** We would like to get all plug pages, selecting only `name` , `author`
and `repo` columns and then sort by last modified time.
**Result:** Okay, this is much better. However, I believe this needs a touch
from a visual perspective.
2023-07-02 09:25:32 +00:00
<!-- #query page select name, author, repo where type = "plug" order by lastModified desc limit 5 -->
2023-08-04 16:56:55 +00:00
|name |author |repo |
2023-07-02 09:25:32 +00:00
|--|--|--|
2023-08-04 16:56:55 +00:00
|🔌 Github |Zef Hemel|https://github.com/silverbulletmd/silverbullet-github |
|🔌 Mattermost|Zef Hemel|https://github.com/silverbulletmd/silverbullet-mattermost|
|🔌 Git |Zef Hemel|https://github.com/silverbulletmd/silverbullet-git |
|🔌 Ghost |Zef Hemel|https://github.com/silverbulletmd/silverbullet-ghost |
|🔌 Share | |https://github.com/silverbulletmd/silverbullet |
2022-10-28 14:17:40 +00:00
<!-- /query -->
#### 6.4 Display the data in a format defined by a template
**Goal:** We would like to display the data from step 5.3 in a nice format using bullet points with links to Plug pages, with the author name and a link to their GitHub repo.
**Result:** Here you go. This is the result we would like to achieve 🎉. Did you see how I used `render` and `template/plug` in a query? 🚀
2023-07-02 09:25:32 +00:00
<!-- #query page where type = "plug" order by lastModified desc limit 5 render [[template/plug]] -->
2023-08-04 16:56:55 +00:00
* [[🔌 Github]] by **Zef Hemel** ([repo](https://github.com/silverbulletmd/silverbullet-github))
* [[🔌 Mattermost]] by **Zef Hemel** ([repo](https://github.com/silverbulletmd/silverbullet-mattermost))
* [[🔌 Git]] by **Zef Hemel** ([repo](https://github.com/silverbulletmd/silverbullet-git))
* [[🔌 Ghost]] by **Zef Hemel** ([repo](https://github.com/silverbulletmd/silverbullet-ghost))
* [[🔌 Share]]
2022-10-28 14:17:40 +00:00
<!-- /query -->
PS: You don't need to select only certain fields to use templates. Templates are
smart enough to get only the information needed to render the data. Therefore,
the following queries are the same in terms of end result when using the
templates.
2022-11-25 15:01:05 +00:00
<!-- #query page select name author repo uri where type = "plug" order by lastModified desc limit 5 render [[template/plug]] -->
2022-10-28 14:17:40 +00:00
2022-11-25 15:01:05 +00:00
and:
<!-- #query page where type = "plug" order by lastModified desc limit 5 render [[template/plug]] -->