wttr.in/README.md

612 lines
21 KiB
Markdown
Raw Normal View History

*wttr.in — the right way to check the weather!*
wttr.in is a console-oriented weather forecast service that supports various information
representation methods like terminal-oriented ANSI-sequences for console HTTP clients
(curl, httpie, or wget), HTML for web browsers, or PNG for graphical viewers.
2017-03-01 16:29:13 +00:00
2017-03-01 17:34:49 +00:00
wttr.in uses [wego](http://github.com/schachmat/wego) for visualization
and various data sources for weather forecast information.
2017-03-01 16:29:13 +00:00
You can see it running here: [wttr.in](http://wttr.in).
2015-12-26 22:22:13 +00:00
2015-12-26 22:24:34 +00:00
## Usage
2015-12-26 22:22:13 +00:00
You can access the service from a shell or from a Web browser like this:
2015-12-26 22:22:13 +00:00
$ curl wttr.in
Weather for City: Paris, France
\ / Clear
2020-03-23 09:09:11 +00:00
.-. 10 11 °C
― ( ) ― ↑ 11 km/h
`- 10 km
/ \ 0.0 mm
2015-12-26 22:22:13 +00:00
Here is an actual weather report for your location (it's live!):
2017-03-01 16:29:13 +00:00
![Weather Report](http://wttr.in/MyLocation.png?)
(It's not your actual location - GitHub's CDN hides your real IP address with its own IP address,
but it's still a live weather report in your language.)
Or in PowerShell:
```PowerShell
Invoke-RestMethod http://wttr.in
```
Want to get the weather information for a specific location? You can add the desired location to the URL in your
request like this:
2015-12-26 22:22:13 +00:00
$ curl wttr.in/London
$ curl wttr.in/Moscow
$ curl wttr.in/Salt+Lake+City
2015-12-26 22:22:13 +00:00
If you omit the location name, you will get the report for your current location based on your IP address.
Use 3-letter airport codes in order to get the weather information at a certain airport:
2015-12-26 22:22:13 +00:00
$ curl wttr.in/muc # Weather for IATA: muc, Munich International Airport, Germany
2016-02-17 16:31:01 +00:00
$ curl wttr.in/ham # Weather for IATA: ham, Hamburg Airport, Germany
2015-12-26 22:22:13 +00:00
Let's say you'd like to get the weather for a geographical location other than a town or city - maybe an attraction
in a city, a mountain name, or some special location. Add the character `~` before the name to look up that special
location name before the weather is then retrieved:
$ curl wttr.in/~Vostok+Station
$ curl wttr.in/~Eiffel+Tower
$ curl wttr.in/~Kilimanjaro
For these examples, you'll see a line below the weather forecast output that shows the geolocation
results of looking up the location:
Location: Vostok Station, станция Восток, AAT, Antarctica [-78.4642714,106.8364678]
Location: Tour Eiffel, 5, Avenue Anatole France, Gros-Caillou, 7e, Paris, Île-de-France, 75007, France [48.8582602,2.29449905432]
2020-03-23 09:09:11 +00:00
Location: Kilimanjaro, Northern, Tanzania [-3.4762789,37.3872648]
You can also use IP-addresses (direct) or domain names (prefixed with `@`) to specify a location:
2015-12-26 22:22:13 +00:00
$ curl wttr.in/@github.com
$ curl wttr.in/@msu.ru
2017-03-01 16:29:13 +00:00
To get detailed information online, you can access the [/:help](http://wttr.in/:help) page:
2015-12-26 22:22:13 +00:00
$ curl wttr.in/:help
### Weather Units
2017-03-01 17:34:49 +00:00
2018-07-18 15:12:32 +00:00
By default the USCS units are used for the queries from the USA and the metric system for the rest of the world.
You can override this behavior by adding `?u` or `?m` to a URL like this:
2017-03-01 17:34:49 +00:00
$ curl wttr.in/Amsterdam?u
$ curl wttr.in/Amsterdam?m
2020-05-10 15:20:52 +00:00
## Supported output formats and views
2017-03-15 16:16:47 +00:00
2020-05-10 15:20:52 +00:00
wttr.in currently supports five output formats:
2017-03-15 16:16:47 +00:00
* ANSI for the terminal;
2020-05-10 15:20:52 +00:00
* Plain-text for the terminal and scripts;
2017-03-15 16:16:47 +00:00
* HTML for the browser;
* PNG for the graphical viewers;
* JSON for scripts and APIs;
* Prometheus metrics for scripts and APIs.
2017-03-15 16:16:47 +00:00
The ANSI and HTML formats are selected basing on the User-Agent string.
The PNG format can be forced by adding `.png` to the end of the query:
$ wget wttr.in/Paris.png
You can use all of the options with the PNG-format like in an URL, but you have
to separate them with `_` instead of `?` and `&`:
$ wget wttr.in/Paris_0tqp_lang=fr.png
Useful options for the PNG format:
2017-03-15 16:16:47 +00:00
* `t` for transparency (`transparency=150`);
* transparency=0..255 for a custom transparency level.
Transparency is a useful feature when weather PNGs are used to add weather data to pictures:
2017-03-15 16:16:47 +00:00
$ convert source.jpg <( curl wttr.in/Oymyakon_tqp0.png ) -geometry +50+50 -composite target.jpg
2017-03-15 16:16:47 +00:00
In this example:
2017-03-15 16:16:47 +00:00
* `source.jpg` - source file;
* `target.jpg` - target file;
* `Oymyakon` - name of the location;
* `tqp0` - options (recommended).
2017-03-15 16:16:47 +00:00
![Picture with weather data](https://pbs.twimg.com/media/C69-wsIW0AAcAD5.jpg)
2019-10-01 13:33:32 +00:00
You can embed a special wttr.in widget, that displays the weather condition for the current or a selected location, into a HTML page using the [wttr-switcher](https://github.com/midzer/wttr-switcher). That is how it looks like: [wttr-switcher-example](https://midzer.github.io/wttr-switcher/) or on a real world web site: https://feuerwehr-eisolzried.de/.
2019-09-19 16:28:21 +00:00
2019-09-19 17:15:08 +00:00
![Embedded wttr.in example at feuerwehr-eisolzried.de](https://user-images.githubusercontent.com/3875145/65265457-50eac180-db11-11e9-8f9b-2e1711dfc436.png)
## One-line output
For one-line output format, specify additional URL parameter `format`:
```
$ curl wttr.in/Nuremberg?format=3
Nuremberg: 🌦 +11⁰C
```
Available preconfigured formats: 1, 2, 3, 4 and the custom format using the percent notation (see below).
2019-01-14 08:25:46 +00:00
2019-06-25 13:44:20 +00:00
You can specify multiple locations separated with `:` (for repeating queries):
```
$ curl wttr.in/Nuremberg:Hamburg:Berlin?format=3
Nuremberg: 🌦 +11⁰C
```
Or to process all this queries at once:
```
2019-02-18 14:02:21 +00:00
$ curl -s 'wttr.in/{Nuremberg,Hamburg,Berlin}?format=3'
Nuremberg: 🌦 +11⁰C
Hamburg: 🌦 +8⁰C
Berlin: 🌦 +8⁰C
```
To specify your own custom output format, use the special `%`-notation:
```
c Weather condition,
C Weather condition textual name,
2019-02-19 19:25:52 +00:00
h Humidity,
t Temperature (Actual),
f Temperature (Feels Like),
w Wind,
l Location,
m Moonphase 🌑🌒🌓🌔🌕🌖🌗🌘,
M Moonday,
p precipitation (mm),
P pressure (hPa),
2020-05-10 21:14:50 +00:00
D Dawn*,
S Sunrise*,
z Zenith*,
s Sunset*,
d Dusk*.
2020-06-26 05:49:47 +00:00
(*times are shown in the local timezone)
```
So, these two calls are the same:
```
$ curl wttr.in/London?format=3
London: ⛅️ +7⁰C
$ curl wttr.in/London?format="%l:+%c+%t\n"
London: ⛅️ +7⁰C
```
Keep in mind, that when using in `tmux.conf`, you have to escape `%` with `%`, i.e. write there `%%` instead of `%`.
2019-08-29 22:01:49 +00:00
In programs, that are querying the service automatically (such as tmux), it is better to use some reasonable update interval. In tmux, you can configure it with `status-interval`.
If several, `:` separated locations, are specified in the query, specify update period
as an additional query parameter `period=`:
```
set -g status-interval 60
WEATHER='#(curl -s wttr.in/London:Stockholm:Moscow\?format\="%%l:+%%c%%20%%t%%60%%w&period=60")'
set -g status-right "$WEATHER ..."
```
![wttr.in in tmux status bar](https://wttr.in/files/example-tmux-status-line.png)
To see emojis in terminal, you need:
1. Terminal support for emojis (was added to Cairo 1.15.8);
2. Font with emojis support.
For the Emoji font, we recommend *Noto Color Emoji*, and a good alternative option would be the *Emoji One* font;
both of them support all necessary emoji glyphs.
Font configuration:
```xml
$ cat ~/.config/fontconfig/fonts.conf
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE fontconfig SYSTEM "fonts.dtd">
<fontconfig>
<alias>
<family>serif</family>
<prefer>
<family>Noto Color Emoji</family>
</prefer>
</alias>
<alias>
<family>sans-serif</family>
<prefer>
<family>Noto Color Emoji</family>
</prefer>
</alias>
<alias>
<family>monospace</family>
<prefer>
<family>Noto Color Emoji</family>
</prefer>
</alias>
</fontconfig>
```
(to apply the configuration, run `fc-cache -f -v`)
2019-08-25 21:25:28 +00:00
## Data-rich output format
2019-08-28 00:45:27 +00:00
In the experimental data-rich output format, that is available under the view code `v2`,
a lot of additional weather and astronomical information is available:
* Temperature, and precepetation changes forecast throughout the days;
* Moonphase for today and the next three days;
* The current weather condition, temperature, humidity, windspeed and direction, pressure;
* Timezone;
* Dawn, sunrise, noon, sunset, dusk time for he selected location;
* Precise geographical coordinates for the selected location.
```
$ curl v2.wttr.in/München
```
or
```
$ curl wttr.in/München?format=v2
```
or, if you prefer Nerd Fonts instead of Emoji, `v2d` (day) or `v2n` (night):
```
$ curl v2d.wttr.in/München
```
![data-reach output format](https://wttr.in/files/example-wttr-v2.png)
(The mode is experimental, and it has several limitations currently:
* It works only in terminal;
* Only English is supported).
2019-08-30 15:46:57 +00:00
Currently, you need some tweaks for some terminals, to get the best possible visualization.
2019-08-28 08:45:16 +00:00
2019-08-30 15:46:57 +00:00
### URXVT
2020-06-28 01:01:33 +00:00
Depending on your configuration you might be taking all steps, or only a few. URXVT currently doesn't support emoji related fonts, but we can get almost the same effect using *Font-Symbola*. So add to your `.Xresources` file the following line:
2019-08-30 15:46:57 +00:00
```
xft:symbola:size=10:minspace=False
```
You can add it _after_ your preferred font and it will only show up when required.
Then, if you see or feel like you're having spacing issues, add this: `URxvt.letterSpace: 0`
For some reason URXVT sometimes stops deciding right the word spacing and we need to force it this way.
The result, should look like:
![URXVT Emoji line](https://user-images.githubusercontent.com/24360204/63842949-1d36d480-c975-11e9-81dd-998d1329bd8a.png)
2019-08-28 08:45:16 +00:00
2020-06-13 11:05:28 +00:00
## Different output formats
2020-05-10 15:20:52 +00:00
2020-06-13 11:05:28 +00:00
### JSON output
The JSON format is a feature providing access to *wttr.in* data through an easy-to-parse format, without requiring the user to create a complex script to reinterpret wttr.in's graphical output.
2020-05-10 15:20:52 +00:00
To fetch information in JSON format, use the following syntax:
$ curl wttr.in/Detroit?format=j1
This will fetch information on the Detroit region in JSON format. The j1 format code is used to allow for the use of other layouts for the JSON output.
The result will look something like the following:
```json
{
"current_condition": [
{
"FeelsLikeC": "25",
"FeelsLikeF": "76",
"cloudcover": "100",
"humidity": "76",
"observation_time": "04:08 PM",
"precipMM": "0.2",
"pressure": "1019",
"temp_C": "22",
"temp_F": "72",
"uvIndex": 5,
"visibility": "16",
"weatherCode": "122",
"weatherDesc": [
{
"value": "Overcast"
}
],
"weatherIconUrl": [
{
"value": ""
}
],
"winddir16Point": "NNE",
"winddirDegree": "20",
"windspeedKmph": "7",
"windspeedMiles": "4"
}
],
...
```
2020-05-10 15:20:52 +00:00
Most of these values are self-explanatory, aside from `weatherCode`. The `weatherCode` is an enumeration which you can find at either [the WorldWeatherOnline website](https://www.worldweatheronline.com/developer/api/docs/weather-icons.aspx) or [in the wttr.in source code](https://github.com/chubin/wttr.in/blob/master/lib/constants.py).
2020-06-13 11:05:28 +00:00
### Prometheus Metrics Output
2020-06-13 11:05:28 +00:00
The [Prometheus](https://github.com/prometheus/prometheus) Metrics format is a feature providing access to *wttr.in* data through an easy-to-parse format for monitoring systems, without requiring the user to create a complex script to reinterpret wttr.in's graphical output.
To fetch information in Prometheus format, use the following syntax:
$ curl wttr.in/Detroit?format=p1
2020-06-13 11:05:28 +00:00
This will fetch information on the Detroit region in Prometheus Metrics format. The `p1` format code is used to allow for the use of other layouts for the Prometheus Metrics output.
A possible configuration for Prometheus could look like this:
```yaml
- job_name: 'wttr_in_detroit'
static_configs:
- targets: ['wttr.in']
metrics_path: '/Detroit'
params:
format: ['p1']
```
The result will look something like the following:
# HELP temperature_feels_like_celsius Feels Like Temperature in Celsius
temperature_feels_like_celsius{forecast="current"} 7
# HELP temperature_feels_like_fahrenheit Feels Like Temperature in Fahrenheit
temperature_feels_like_fahrenheit{forecast="current"} 45
2020-06-13 11:05:28 +00:00
[truncated]
...
2020-05-10 15:20:52 +00:00
## Moon phases
2017-03-01 16:29:13 +00:00
2020-03-23 11:18:36 +00:00
wttr.in can also be used to check the phase of the Moon. This example shows how to see the current Moon phase
in the full-output mode:
2017-03-01 16:29:13 +00:00
$ curl wttr.in/Moon
Get the Moon phase for a particular date by adding `@YYYY-MM-DD`:
2017-03-01 16:29:13 +00:00
$ curl wttr.in/Moon@2016-12-25
The Moon phase information uses [pyphoon](https://github.com/chubin/pyphoon) as its backend.
2017-03-01 16:29:13 +00:00
2020-03-23 11:18:36 +00:00
To get the moon phase information in the online mode, use `%m`:
$ curl wttr.in/London?format=%m
🌖
2020-08-17 02:09:55 +00:00
Keep in mind that the Unicode representation of moonphases suffers 2 caveats:
2020-03-23 09:09:11 +00:00
- With some fonts, the representation `🌘` is ambiguous, for it either seem
2020-06-28 01:01:33 +00:00
almost-shadowed or almost-lit, depending on whether your terminal is in
2020-03-23 09:09:11 +00:00
light mode or dark mode. Relying on colored fonts like `noto-fonts` works
around this problem.
- The representation `🌘` is also ambiguous, for it means "last quarter" in
northern hemisphere, but "first quarter" in souther hemisphere. It also means
nothing in tropical zones. This is a limitation that
[Unicode](https://www.unicode.org/L2/L2017/17304-moon-var.pdf) is aware about.
But it has not been worked around at `wttr.in` yet.
2020-03-23 11:18:36 +00:00
See #247, #364 for the corresponding tracking issues,
and [pyphoon#1](https://github.com/chubin/pyphoon/issues/1) for pyphoon. Any help is welcome.
2020-03-23 09:09:11 +00:00
## Internationalization and localization
wttr.in supports multilingual locations names that can be specified in any language in the world
(it may be surprising, but many locations in the world don't have an English name).
2017-03-01 16:29:13 +00:00
The query string should be specified in Unicode (hex-encoded or not). Spaces in the query string
must be replaced with `+`:
2017-03-01 16:29:13 +00:00
$ curl wttr.in/станция+Восток
Weather report: станция Восток
Overcast
.--. -65 -47 °C
.-( ). ↑ 23 km/h
(___.__)__) 15 km
0.0 mm
The language used for the output (except the location name) does not depend on the input language
and it is either English (by default) or the preferred language of the browser (if the query
was issued from a browser) that is specified in the query headers (`Accept-Language`).
The language can be set explicitly when using console clients by using command-line options like this:
curl -H "Accept-Language: fr" wttr.in
http GET wttr.in Accept-Language:ru
2017-03-01 16:29:13 +00:00
The preferred language can be forced using the `lang` option:
$ curl wttr.in/Berlin?lang=de
The third option is to choose the language using the DNS name used in the query:
$ curl de.wttr.in/Berlin
wttr.in is currently translated into 54 languages, and the number of supported languages is constantly growing.
2020-03-23 09:09:11 +00:00
See [/:translation](http://wttr.in/:translation) to learn more about the translation process,
to see the list of supported languages and contributors, or to know how you can help to translate wttr.in
in your language.
2017-03-01 16:29:13 +00:00
![Queries to wttr.in in various languages](https://pbs.twimg.com/media/C7hShiDXQAES6z1.jpg)
## Windows Users
There are currently two Windows related issues that prevent the examples found on this page from working exactly as expected out of the box. Until Microsoft fixes the issues, there are a few workarounds. To circumvent both issues you may use a shell such as `bash` on the [Windows Subsystem for Linux (WSL)](https://docs.microsoft.com/en-us/windows/wsl/install-win10) or read on for alternative solutions.
### Garbage characters in the output
There is a limitation of the current Win32 version of `curl`. Until the [Win32 curl issue](https://github.com/chubin/wttr.in/issues/18#issuecomment-474145551) is resolved and rolled out in a future Windows release, it is recommended that you use Powershells `Invoke-Web-Request` command instead:
- `(Invoke-WebRequest http://wttr.in).Content`
### Missing or double wide diagonal wind direction characters
The second issue is regarding the width of the diagonal arrow glyphs that some Windows Terminal Applications such as the default `conhost.exe` use. At the time of writing this, `ConEmu.exe`, `ConEmu64.exe` and Terminal Applications built on top of ConEmu such as Cmder (`cmder.exe`) use these double-wide glyphs by default. The result is the same with all of these programs, either a missing character for certain wind directions or a broken table in the output or both. Some third-party Terminal Applications have addressed the wind direction glyph issue but that fix depends on the font and the Terminal Application you are using.
One way to display the diagonal wind direction glyphs in your Terminal Application is to use [Windows Terminal](https://www.microsoft.com/en-us/p/windows-terminal-preview/9n0dx20hk701?activetab=pivot:overviewtab) which is currently available in the [Microsoft Store](https://www.microsoft.com/en-us/p/windows-terminal-preview/9n0dx20hk701?activetab=pivot:overviewtab). Windows Terminal is currently a preview release and will be rolled out as the default Terminal Application in an upcoming release. If your output is still skewed after using Windows Terminal then try maximizing the terminal window.
Another way you can display the diagonal wind direction is to swap out the problematic characters with forward and backward slashes as shown [here](https://github.com/chubin/wttr.in/issues/18#issuecomment-405640892).
2020-03-23 09:09:11 +00:00
## Installation
2015-12-26 22:22:13 +00:00
To install the application:
2015-12-26 22:22:13 +00:00
2016-02-17 12:07:16 +00:00
1. Install external dependencies
2. Install Python dependencies used by the service
2019-05-09 15:40:22 +00:00
3. Configure IP2Location (optional)
4. Get a WorldWeatherOnline API and configure wego
2015-12-26 22:22:13 +00:00
5. Configure wttr.in
6. Configure the HTTP-frontend service
2015-12-26 22:22:13 +00:00
### Install external dependencies
wttr.in has the following external dependencies:
2015-12-26 22:22:13 +00:00
* [golang](https://golang.org/doc/install), wego dependency
2016-03-30 16:17:57 +00:00
* [wego](https://github.com/schachmat/wego), weather client for terminal
2015-12-26 22:22:13 +00:00
After you install [golang](https://golang.org/doc/install), install `wego`:
2015-12-26 22:22:13 +00:00
2017-04-17 01:54:45 +00:00
$ go get -u github.com/schachmat/wego
$ go install github.com/schachmat/wego
2015-12-26 22:22:13 +00:00
### Install Python dependencies
2015-12-26 22:22:13 +00:00
Python requirements:
* Flask
* geoip2
* geopy
* requests
* gevent
If you want to get weather reports as PNG files, you'll also need to install:
* PIL
* pyte (>=0.6)
* necessary fonts
2020-03-23 09:09:11 +00:00
You can install most of them using `pip`.
2015-12-26 22:22:13 +00:00
Some python package use LLVM, so install it first:
$ apt-get install llvm-7 llvm-7-dev
2015-12-26 22:22:13 +00:00
If `virtualenv` is used:
$ virtualenv -p python3 ve
$ ve/bin/pip3 install -r requirements.txt
$ ve/bin/python3 bin/srv.py
2015-12-26 22:22:13 +00:00
Also, you need to install the geoip2 database.
You can use a free database GeoLite2 that can be downloaded from (http://dev.maxmind.com/geoip/geoip2/geolite2/).
2015-12-26 22:22:13 +00:00
2019-05-09 15:40:22 +00:00
### Configure IP2Location (optional)
If you want to use the IP2location service for IP-addresses that are not covered by GeoLite2,
you have to obtain a API key of that service, and after that save into the `~/.ip2location.key` file:
```
$ echo 'YOUR_IP2LOCATION_KEY' > ~/.ip2location.key
```
If you don't have this file, the service will be silently skipped (it is not a big problem,
because the MaxMind database is pretty good).
### Installation with Docker
* Install Docker
* Build Docker Image
* These files should be mounted by the user at runtime:
```
/root/.wegorc
/root/.ip2location.key (optional)
/app/airports.dat
/app/GeoLite2-City.mmdb
```
2019-05-09 15:40:22 +00:00
### Get a WorldWeatherOnline key and configure wego
2015-12-26 22:22:13 +00:00
To get a WorldWeatherOnline API key, you must register here:
2020-03-23 09:09:11 +00:00
2015-12-26 22:22:13 +00:00
https://developer.worldweatheronline.com/auth/register
After you have a WorldWeatherOnline key, you can save it into the
WWO key file: `~/.wwo.key`
Also, you have to specify the key in the `wego` configuration:
2015-12-26 22:22:13 +00:00
```json
$ cat ~/.wegorc
{
"APIKey": "00XXXXXXXXXXXXXXXXXXXXXXXXXXX",
"City": "London",
"Numdays": 3,
"Imperial": false,
"Lang": "en"
}
```
2015-12-26 22:22:13 +00:00
The `City` parameter in `~/.wegorc` is ignored.
### Configure wttr.in
Configure the following environment variables that define the path to the local `wttr.in`
installation, to the GeoLite database, and to the `wego` installation. For example:
2015-12-26 22:22:13 +00:00
```bash
export WTTR_MYDIR="/home/igor/wttr.in"
export WTTR_GEOLITE="/home/igor/wttr.in/GeoLite2-City.mmdb"
export WTTR_WEGO="/home/igor/go/bin/wego"
export WTTR_LISTEN_HOST="0.0.0.0"
export WTTR_LISTEN_PORT="8002"
```
2015-12-26 22:22:13 +00:00
### Configure the HTTP-frontend service
2015-12-26 22:22:13 +00:00
It's recommended that you also configure the web server that will be used to access the service:
2015-12-26 22:22:13 +00:00
```nginx
server {
listen [::]:80;
server_name wttr.in *.wttr.in;
access_log /var/log/nginx/wttr.in-access.log main;
error_log /var/log/nginx/wttr.in-error.log;
2015-12-26 22:22:13 +00:00
location / {
proxy_pass http://127.0.0.1:8002;
2015-12-26 22:22:13 +00:00
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $remote_addr;
2015-12-26 22:22:13 +00:00
client_max_body_size 10m;
client_body_buffer_size 128k;
2015-12-26 22:22:13 +00:00
proxy_connect_timeout 90;
proxy_send_timeout 90;
proxy_read_timeout 90;
2015-12-26 22:22:13 +00:00
proxy_buffer_size 4k;
proxy_buffers 4 32k;
proxy_busy_buffers_size 64k;
proxy_temp_file_write_size 64k;
2015-12-26 22:22:13 +00:00
expires off;
}
}
```