Your browser lacks required capabilities. Please upgrade it or switch to another to continue.
Loading…
<<widget "newinfo">>
<<if !$knowledge.has($args[0])>>
<div class="newinfo">
<h2> you've found new information!! </h2>
<<run $knowledge.add($args[0])>>
<p>
Adding <strong>"<<print $args[0]>>"</strong> to your "What you know" list.
</p>
</div>
<</if>>
<</widget>>
<<widget "init">>
<<if ndef $knowledge>>
<<set $knowledge = new Set()>>
<</if>>
<</widget>>
<!-- <<ask "Why do you think this happened?" "$variable" "Next page">> -->
<<widget "ask">>
<div style="margin-top: 3em;">
<<print $args[0]>> <<textbox $args[1] "" $args[2]>>
<div style="display: flex; margin-top: .5em;">
<<link "Submit" $args[2]>><</link>>
<<link "I don't know" $args[2]>><</link>>
</div>
</div>
<</widget>>
<<widget "ask-area">>
<div style="margin-top: 3em;">
<<print $args[0]>> <<textarea $args[1] "" $args[2]>>
<div style="display: flex; margin-top: .5em;">
<<link "Submit" $args[2]>><</link>>
<<link "I don't know" $args[2]>><</link>>
</div>
</div>
<</widget>>
<!-- <<you-said "Your reason for the SYN packets being retried:" $why_syn>> -->
<<widget "you-said">>
<p>
<<if !$args[1] eq "">><<print $args[0]>> "<strong><<print $args[1]>></strong>".<</if>>
</p>
<</widget>><div style="text-align: left">
<strong>Files:</strong>
<p style="font-family: monospace; padding-left: 1em;">
<<link "code.py">>
<<script>>
Dialog.setup("code.py");
Dialog.wiki(`
<html>
<pre>
import redis <br>
import flask <br>
<br>
app = flask.Flask(__name__) <br>
r = redis.Redis(host='localhost', port=36379, db=0) <br>
<br>
@app.route('/count') <br>
def count(): <br>
r.incr('hits') <br>
return 'This page has been viewed %s times.' % r.get('hits') <br>
<br>
app.run(host='0.0.0.0', port=3000)
</pre>
</html>
`);
Dialog.open();
<</script>>
<</link>>
<br>
<<link "docker-compose.yaml">>
<<script>>
Dialog.setup("docker-compose.yaml");
Dialog.wiki(`
<html>
<pre>
version: "3" <br>
services: <br>
backend: <br>
build: '.' <br>
ports: <br>
- "3000:3000" <br>
redis: <br>
image: redis <br>
container_name: cache <br>
ports: <br>
- 36379:6379 <br>
restart: always
</pre>
</html>
`);
Dialog.open();
<</script>>
<</link>>
</p>
<strong>What you know:</strong>
<<if $knowledge.size eq 0>> nothing yet!
<<else>>
<ol> <<for _i, _name range $knowledge>> <li> _name</li> <</for>> </ol>
<</if>>
</div>
<div style="margin-top: 4em">
<a href="/"><< Back to all puzzles</a>
</div><<init>>
<p>
Hello! This is a mystery where your goal is to solve a debugging problem!
You'll collect clues, interpret evidence, and ultimately solve the Case of the
Failed Docker Connection!
</p>
<p>
In the sidebar, you'll notice a "What you know" list. As you collect more
clues, the list will update with what you've learned.
</p>
<p>
Click "Start" to get started.
</p>
[[Start->Landing]]<p>
You're working on a Python web service, and you need a local cache for some
data. You decide to use Redis as a cache, and so you set up a Redis container
in your local environment with Docker Compose.
</p>
<p>
But when you add the Redis code and try to load a page from your server, you get a
500 internal server error. Oh no!
</p>
[[Read the error logs -> Error]]This is the error message your Python service prints out every time you make a request:
<pre>
backend_1 | redis.exceptions.ConnectionError: Error 99 connecting <br>
to localhost:36379. Cannot assign requested address.
</pre>
That's useful -- we've learned something already!
<<newinfo "The Python program can't connect to localhost:36379">>
<div class="nav">
[[Read the full error logs -> Read more logs]]
[[Look at your Python code -> Python program]]
[[I know how to fix the bug -> Guess answer beginning]]
</div><html>
<pre>
backend_1 | [2021-11-08 21:30:44,180] ERROR in app: Exception on /count [GET] <br>
backend_1 | Traceback (most recent call last): <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/connection.py", line 559, in connect <br>
backend_1 | sock = self._connect() <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/connection.py", line 615, in _connect <br>
backend_1 | raise err <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/connection.py", line 603, in _connect <br>
backend_1 | sock.connect(socket_address) <br>
backend_1 | OSError: [Errno 99] Cannot assign requested address <br>
backend_1 | <br>
backend_1 | During handling of the above exception, another exception occurred: <br>
backend_1 | <br>
backend_1 | Traceback (most recent call last): <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/flask/app.py", line 2073, in wsgi_app <br>
backend_1 | response = self.full_dispatch_request() <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/flask/app.py", line 1518, in full_dispatch_request <br>
backend_1 | rv = self.handle_user_exception(e) <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/flask/app.py", line 1516, in full_dispatch_request <br>
backend_1 | rv = self.dispatch_request() <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/flask/app.py", line 1502, in dispatch_request <br>
backend_1 | return self.ensure_sync(self.view_functions[rule.endpoint])(**req.view_args) <br>
backend_1 | File "/code.py", line 10, in count <br>
backend_1 | r.incr('hits') <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/client.py", line 1641, in incr <br>
backend_1 | return self.incrby(name, amount) <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/client.py", line 1650, in incrby <br>
backend_1 | return self.execute_command('INCRBY', name, amount) <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/client.py", line 898, in execute_command <br>
backend_1 | conn = self.connection or pool.get_connection(command_name, **options) <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/connection.py", line 1192, in get_connection <br>
backend_1 | connection.connect() <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/connection.py", line 563, in connect <br>
backend_1 | raise ConnectionError(self._error_message(e)) <br>
backend_1 | redis.exceptions.ConnectionError: Error 99 connecting to localhost:36379. Cannot assign requested address. <br>
backend_1 | 172.19.0.1 - - [08/Nov/2021 21:30:44] "GET /count HTTP/1.1" 500 - <br>
backend_1 | [2021-11-08 21:48:49,138] ERROR in app: Exception on /count [GET] <br>
backend_1 | Traceback (most recent call last): <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/connection.py", line 559, in connect <br>
backend_1 | sock = self._connect() <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/connection.py", line 615, in _connect <br>
backend_1 | raise err <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/connection.py", line 603, in _connect <br>
backend_1 | sock.connect(socket_address) <br>
backend_1 | OSError: [Errno 99] Cannot assign requested address <br>
backend_1 | <br>
backend_1 | During handling of the above exception, another exception occurred: <br>
backend_1 | <br>
backend_1 | Traceback (most recent call last): <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/flask/app.py", line 2073, in wsgi_app <br>
backend_1 | response = self.full_dispatch_request() <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/flask/app.py", line 1518, in full_dispatch_request <br>
backend_1 | rv = self.handle_user_exception(e) <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/flask/app.py", line 1516, in full_dispatch_request <br>
backend_1 | rv = self.dispatch_request() <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/flask/app.py", line 1502, in dispatch_request <br>
backend_1 | return self.ensure_sync(self.view_functions[rule.endpoint])(**req.view_args) <br>
backend_1 | File "/code.py", line 10, in count <br>
backend_1 | r.incr('hits') <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/client.py", line 1641, in incr <br>
backend_1 | return self.incrby(name, amount) <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/client.py", line 1650, in incrby <br>
backend_1 | return self.execute_command('INCRBY', name, amount) <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/client.py", line 898, in execute_command <br>
backend_1 | conn = self.connection or pool.get_connection(command_name, **options) <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/connection.py", line 1192, in get_connection <br>
backend_1 | connection.connect() <br>
backend_1 | File "/usr/local/lib/python3.8/dist-packages/redis/connection.py", line 563, in connect <br>
backend_1 | raise ConnectionError(self._error_message(e)) <br>
backend_1 | redis.exceptions.ConnectionError: Error 99 connecting to localhost:36379. Cannot assign requested address. <br>
backend_1 | 172.19.0.1 - - [08/Nov/2021 21:48:49] "GET /count HTTP/1.1" 500 - <br>
</pre>
</html>
Do you think there's any more information about the bug here?
<div class="nav">
[[Yes -> Long error logs 2]]
[[No -> Long error logs 2]]
</div>The error logs have a lot in them, but the most important thing is that it
can't connect to <code>localhost:36379</code> -- there's nothing else relevant.
<div class="nav">
[[Go back -> Error]]
</div><p>
Here's your Python program:
</p>
<html>
<pre>
import redis <br>
import flask <br>
<br>
app = flask.Flask(__name__) <br>
r = redis.Redis(host='localhost', port=36379, db=0) <br>
<br>
@app.route('/count') <br>
def count(): <br>
r.incr('hits') <br>
return 'This page has been viewed %s times.' % r.get('hits') <br>
<br>
app.run(host='0.0.0.0', port=3000) <br>
</pre>
</html>
The reason this is failing is pretty clear -- this line of code:
<html>
<pre>
r = redis.Redis(host='localhost', port=36379, db=0)
</pre>
</html>
<p>
is trying to connect to <code>localhost:36379</code>, but Redis isn't running on that port.
</p>
<p>
Next, we need to figure out why Redis isn't running on <code>localhost:36379</code>! What do you want to try first?
</p>
<div class="nav">
[[Read docker-compose.yaml -> docker-compose yaml]]
[[Check if Redis is running -> check redis running]]
[[Ping the container -> ping container]]
</div>You decide to check if Redis is running.
<<ask "How can you find out if Redis is running?" "$redis" "redis running 2">><<you-said "You said:" $redis>>
This game isn't very smart, so it only knows about 3 options. Hopefully one of these is what you said:
<div class="nav">
[[use the Redis CLI -> redis-cli]]
[[use telnet -> telnet]]
[[look at the logs -> look at logs]]
</div>You run <code>docker-compose logs redis</code> to look at the logs. Here's what you see:
<html>
<pre>
Attaching to cache <br>
08 Nov 2021 20:27:21.946 # oO0OoO0OoO0Oo Redis is starting oO0OoO0OoO0Oo <br>
08 Nov 2021 20:27:21.946 # Redis version=6.2.6, bits=64, commit=00000000, modified=0, pid=1, just started <br>
08 Nov 2021 20:27:21.946 # Warning: no config file specified, using the default config. In order to specify a config file use redis-server /path/to/redis.conf <br>
08 Nov 2021 20:27:21.946 * monotonic clock: POSIX clock_gettime <br>
08 Nov 2021 20:27:21.946 * Running mode=standalone, port=6379. <br>
08 Nov 2021 20:27:21.947 # Server initialized <br>
08 Nov 2021 20:27:21.947 # WARNING overcommit_memory is set to 0! Background save may fail under low memory condition. To fix this issue add 'vm.overcommit_memory = 1' to /etc/sysctl.conf and then reboot or run the command 'sysctl vm.overcommit_memory=1' for this to take effect. <br>
08 Nov 2021 20:27:21.947 * Ready to accept connections <br>
gnal-handler (1636403244) Received SIGTERM scheduling shutdown... <br>
08 Nov 2021 20:27:24.355 # User requested shutdown... <br>
08 Nov 2021 20:27:24.355 * Saving the final RDB snapshot before exiting. <br>
08 Nov 2021 20:27:24.356 * DB saved on disk <br>
08 Nov 2021 20:27:24.356 # Redis is now ready to exit, bye bye... <br>
08 Nov 2021 20:27:27.078 # oO0OoO0OoO0Oo Redis is starting oO0OoO0OoO0Oo <br>
08 Nov 2021 20:27:27.079 # Redis version=6.2.6, bits=64, commit=00000000, modified=0, pid=1, just started <br>
08 Nov 2021 20:27:27.079 # Warning: no config file specified, using the default config. In order to specify a config file use redis-server /path/to/redis.conf <br>
08 Nov 2021 20:27:27.079 * monotonic clock: POSIX clock_gettime <br>
08 Nov 2021 20:27:27.079 * Running mode=standalone, port=6379. <br>
08 Nov 2021 20:27:27.080 # Server initialized <br>
08 Nov 2021 20:27:27.080 # WARNING overcommit_memory is set to 0! Background save may fail under low memory condition. To fix this issue add 'vm.overcommit_memory = 1' to /etc/sysctl.conf and then reboot or run the command 'sysctl vm.overcommit_memory=1' for this to take effect. <br>
08 Nov 2021 20:27:27.080 * Loading RDB produced by version 6.2.6 <br>
08 Nov 2021 20:27:27.080 * RDB age 3 seconds <br>
08 Nov 2021 20:27:27.080 * RDB memory usage when created 0.77 Mb <br>
08 Nov 2021 20:27:27.080 # Done loading RDB, keys loaded: 0, keys expired: 0. <br>
08 Nov 2021 20:27:27.080 * DB loaded from disk: 0.000 seconds <br>
08 Nov 2021 20:27:27.080 * Ready to accept connections <br>
gnal-handler (1636403286) Received SIGTERM scheduling shutdown... <br>
08 Nov 2021 20:28:06.179 # User requested shutdown... <br>
08 Nov 2021 20:28:06.179 * Saving the final RDB snapshot before exiting. <br>
08 Nov 2021 20:28:06.184 * DB saved on disk <br>
08 Nov 2021 20:28:06.184 # Redis is now ready to exit, bye bye... <br>
08 Nov 2021 20:28:08.294 # oO0OoO0OoO0Oo Redis is starting oO0OoO0OoO0Oo <br>
08 Nov 2021 20:28:08.294 # Redis version=6.2.6, bits=64, commit=00000000, modified=0, pid=1, just started <br>
08 Nov 2021 20:28:08.294 # Warning: no config file specified, using the default config. In order to specify a config file use redis-server /path/to/redis.conf <br>
08 Nov 2021 20:28:08.295 * monotonic clock: POSIX clock_gettime <br>
08 Nov 2021 20:28:08.295 * Running mode=standalone, port=6379. <br>
08 Nov 2021 20:28:08.295 # Server initialized <br>
08 Nov 2021 20:28:08.296 # WARNING overcommit_memory is set to 0! Background save may fail under low memory condition. To fix this issue add 'vm.overcommit_memory = 1' to /etc/sysctl.conf and then reboot or run the command 'sysctl vm.overcommit_memory=1' for this to take effect. <br>
08 Nov 2021 20:28:08.296 * Loading RDB produced by version 6.2.6 <br>
08 Nov 2021 20:28:08.296 * RDB age 2 seconds <br>
08 Nov 2021 20:28:08.296 * RDB memory usage when created 0.77 Mb <br>
08 Nov 2021 20:28:08.296 # Done loading RDB, keys loaded: 0, keys expired: 0. <br>
08 Nov 2021 20:28:08.296 * DB loaded from disk: 0.000 seconds <br>
08 Nov 2021 20:28:08.296 * Ready to accept connections <br>
gnal-handler (1636403298) Received SIGTERM scheduling shutdown... <br>
08 Nov 2021 20:28:18.221 # User requested shutdown... <br>
08 Nov 2021 20:28:18.221 * Saving the final RDB snapshot before exiting. <br>
08 Nov 2021 20:28:18.223 * DB saved on disk <br>
08 Nov 2021 20:28:18.224 # Redis is now ready to exit, bye bye... <br>
08 Nov 2021 20:28:29.015 # oO0OoO0OoO0Oo Redis is starting oO0OoO0OoO0Oo <br>
08 Nov 2021 20:28:29.015 # Redis version=6.2.6, bits=64, commit=00000000, modified=0, pid=1, just started <br>
08 Nov 2021 20:28:29.015 # Warning: no config file specified, using the default config. In order to specify a config file use redis-server /path/to/redis.conf <br>
08 Nov 2021 20:28:29.015 * monotonic clock: POSIX clock_gettime <br>
08 Nov 2021 20:28:29.016 * Running mode=standalone, port=6379. <br>
08 Nov 2021 20:28:29.016 # Server initialized <br>
08 Nov 2021 20:28:29.016 # WARNING overcommit_memory is set to 0! Background save may fail under low memory condition. To fix this issue add 'vm.overcommit_memory = 1' to /etc/sysctl.conf and then reboot or run the command 'sysctl vm.overcommit_memory=1' for this to take effect. <br>
08 Nov 2021 20:28:29.016 * Loading RDB produced by version 6.2.6 <br>
08 Nov 2021 20:28:29.016 * RDB age 11 seconds <br>
08 Nov 2021 20:28:29.016 * RDB memory usage when created 0.77 Mb <br>
08 Nov 2021 20:28:29.016 # Done loading RDB, keys loaded: 0, keys expired: 0. <br>
08 Nov 2021 20:28:29.016 * DB loaded from disk: 0.000 seconds <br>
08 Nov 2021 20:28:29.016 * Ready to accept connections <br>
08 Nov 2021 21:26:54.028 # oO0OoO0OoO0Oo Redis is starting oO0OoO0OoO0Oo <br>
08 Nov 2021 21:26:54.028 # Redis version=6.2.6, bits=64, commit=00000000, modified=0, pid=1, just started <br>
08 Nov 2021 21:26:54.028 # Warning: no config file specified, using the default config. In order to specify a config file use redis-server /path/to/redis.conf <br>
08 Nov 2021 21:26:54.029 * monotonic clock: POSIX clock_gettime <br>
08 Nov 2021 21:26:54.029 * Running mode=standalone, port=6379. <br>
08 Nov 2021 21:26:54.029 # Server initialized <br>
08 Nov 2021 21:26:54.029 # WARNING overcommit_memory is set to 0! Background save may fail under low memory condition. To fix this issue add 'vm.overcommit_memory = 1' to /etc/sysctl.conf and then reboot or run the command 'sysctl vm.overcommit_memory=1' for this to take effect. <br>
08 Nov 2021 21:26:54.030 * Loading RDB produced by version 6.2.6 <br>
08 Nov 2021 21:26:54.030 * RDB age 3516 seconds <br>
08 Nov 2021 21:26:54.030 * RDB memory usage when created 0.77 Mb <br>
08 Nov 2021 21:26:54.030 # Done loading RDB, keys loaded: 0, keys expired: 0. <br>
08 Nov 2021 21:26:54.030 * DB loaded from disk: 0.000 seconds <br>
08 Nov 2021 21:26:54.030 * Ready to accept connections <br>
08 Nov 2021 21:28:04.403 # oO0OoO0OoO0Oo Redis is starting oO0OoO0OoO0Oo <br>
08 Nov 2021 21:28:04.403 # Redis version=6.2.6, bits=64, commit=00000000, modified=0, pid=1, just started <br>
08 Nov 2021 21:28:04.403 # Warning: no config file specified, using the default config. In order to specify a config file use redis-server /path/to/redis.conf <br>
08 Nov 2021 21:28:04.404 * monotonic clock: POSIX clock_gettime <br>
08 Nov 2021 21:28:04.404 * Running mode=standalone, port=6379. <br>
08 Nov 2021 21:28:04.404 # Server initialized <br>
08 Nov 2021 21:28:04.404 # WARNING overcommit_memory is set to 0! Background save may fail under low memory condition. To fix this issue add 'vm.overcommit_memory = 1' to /etc/sysctl.conf and then reboot or run the command 'sysctl vm.overcommit_memory=1' for this to take effect. <br>
08 Nov 2021 21:28:04.404 * Loading RDB produced by version 6.2.6 <br>
08 Nov 2021 21:28:04.405 * RDB age 3586 seconds <br>
08 Nov 2021 21:28:04.405 * RDB memory usage when created 0.77 Mb <br>
08 Nov 2021 21:28:04.405 # Done loading RDB, keys loaded: 0, keys expired: 0. <br>
08 Nov 2021 21:28:04.405 * DB loaded from disk: 0.000 seconds <br>
08 Nov 2021 21:28:04.405 * Ready to accept connections <br>
08 Nov 2021 21:28:27.646 # oO0OoO0OoO0Oo Redis is starting oO0OoO0OoO0Oo <br>
08 Nov 2021 21:28:27.646 # Redis version=6.2.6, bits=64, commit=00000000, modified=0, pid=1, just started <br>
08 Nov 2021 21:28:27.646 # Warning: no config file specified, using the default config. In order to specify a config file use redis-server /path/to/redis.conf <br>
08 Nov 2021 21:28:27.646 * monotonic clock: POSIX clock_gettime <br>
08 Nov 2021 21:28:27.647 * Running mode=standalone, port=6379. <br>
08 Nov 2021 21:28:27.647 # Server initialized <br>
08 Nov 2021 21:28:27.647 # WARNING overcommit_memory is set to 0! Background save may fail under low memory condition. To fix this issue add 'vm.overcommit_memory = 1' to /etc/sysctl.conf and then reboot or run the command 'sysctl vm.overcommit_memory=1' for this to take effect. <br>
08 Nov 2021 21:28:27.647 * Loading RDB produced by version 6.2.6 <br>
08 Nov 2021 21:28:27.647 * RDB age 3609 seconds <br>
08 Nov 2021 21:28:27.647 * RDB memory usage when created 0.77 Mb <br>
08 Nov 2021 21:28:27.647 # Done loading RDB, keys loaded: 0, keys expired: 0. <br>
08 Nov 2021 21:28:27.647 * DB loaded from disk: 0.000 seconds <br>
08 Nov 2021 21:28:27.647 * Ready to accept connections <br>
09 Nov 2021 22:10:54.715 # oO0OoO0OoO0Oo Redis is starting oO0OoO0OoO0Oo <br>
09 Nov 2021 22:10:54.716 # Redis version=6.2.6, bits=64, commit=00000000, modified=0, pid=1, just started <br>
09 Nov 2021 22:10:54.716 # Warning: no config file specified, using the default config. In order to specify a config file use redis-server /path/to/redis.conf <br>
09 Nov 2021 22:10:54.717 * monotonic clock: POSIX clock_gettime <br>
09 Nov 2021 22:10:54.718 * Running mode=standalone, port=6379. <br>
09 Nov 2021 22:10:54.718 # Server initialized <br>
09 Nov 2021 22:10:54.718 # WARNING overcommit_memory is set to 0! Background save may fail under low memory condition. To fix this issue add 'vm.overcommit_memory = 1' to /etc/sysctl.conf and then reboot or run the command 'sysctl vm.overcommit_memory=1' for this to take effect. <br>
09 Nov 2021 22:10:54.719 * Loading RDB produced by version 6.2.6 <br>
09 Nov 2021 22:10:54.719 * RDB age 92556 seconds <br>
09 Nov 2021 22:10:54.719 * RDB memory usage when created 0.77 Mb <br>
09 Nov 2021 22:10:54.719 # Done loading RDB, keys loaded: 0, keys expired: 0. <br>
09 Nov 2021 22:10:54.719 * DB loaded from disk: 0.000 seconds <br>
09 Nov 2021 22:10:54.719 * Ready to accept connections
</pre>
</html>
What do you think? Is Redis running?
<div class="nav">
[[Yes -> logs 2]]
[[No -> logs 2]]
</div>The logs look pretty good, but you still don't feel 100% sure Redis is running.
You'd prefer to actually connect to the Redis service to be sure.
<div class="nav">
[[Go back -> redis running 2]]
</div><p>
You run <code>redis-cli</code> to connect to the Redis instance at <code>localhost:36379</code> Here's
what you see:
</p>
<html>
<pre>
$ redis-cli -h localhost -p 36379 keys "*" <br>
<br>
(empty array)
</pre>
</html>
<p>
What do you think? Does this mean Redis is running?
</p>
<div class="nav">
[[Yes -> redis-cli 2]]
[[No -> redis-cli 2]]
</div><p>
The <code>(empty array)</code> output means that we successfully connected to
Redis, but that the Redis instance is empty. So, it worked!
</p>
<<newinfo "You <strong>can</strong> connect to Redis at localhost:36379 from the command line">>
Now you have a mystery on your hands -- why does connecting to
<code>localhost:36379</code> work when you do it on the command line, but not
from the Python server?
<<ask "What can we do to find out more about this?" "$investigate" "investigate">>You run <code>telnet</code> to connect to <code>localhost:36379</code> Here's what you see:
<html>
<pre>
$ telnet localhost 36379 <br>
Trying ::1... <br>
Connected to localhost. <br>
Escape character is '^]'.
</pre>
</html>
What does this mean? Is Redis running?
<p>
[[Yes -> telnet 2]]
[[No -> telnet 2]]
[[I don't know -> telnet 2]]
</p>This telnet output means that <i>something</i> is running on
<code>localhost:36379</code>, and it's definitely Redis -- you've never used
that port for anything other than Redis.
<<newinfo "You <strong>can</strong> connect to Redis at localhost:36379 from the command line">>
Now you have a mystery on your hands -- why does connecting to
<code>localhost:36379</code> work when you do it on the command line, but not
from the Python server?
<<ask "What can we do to find out more about this?" "$investigate" "investigate">><<you-said "Your idea for how to investigate:" $investigate>>
<p>
We tried connecting to <code>localhost:36379</code> and it worked. But we
didn't try connecting to <code>localhost:36379</code> from <code>inside</code>
the Python Docker container.
</p>
<p>
Let's try that!
</p>
<<ask "How can we run commands inside the Docker container?" "$inside" "investigate 2">><<you-said "You said" $inside>>
<p>
You start a shell inside the Python container with <code>docker-compose exec backend /bin/bash</code>.
</p>
<html>
<pre>
$ docker-compose exec backend /bin/bash <br>
root@31762eeecfda:/#
</pre>
</html>
<p>
Great, that worked! You use the shell to run <code>redis-cli</code> inside the Python container.
</p>
<html>
<pre>
root@31762eeecfda:/# redis-cli -h localhost -p 36379 keys "*" <br>
bash: redis-cli: command not found <br>
root@31762eeecfda:/#
</pre>
</html>
<p>
Oops, the Redis CLI isn't installed inside the Python container.
</p>
<div class="nav">
[[Install the Redis CLI -> investigate 3]]
</div><p>
You're in an Ubuntu container, so this is pretty straightforward to fix -- you install the Redis CLI with <code> apt-get update && apt-get install redis </code>.
</p>
Once that's done, you can run it.
<html>
<pre>
root@31762eeecfda:/# redis-cli -h localhost -p 36379 keys "*" <br>
Could not connect to Redis at localhost:36379: Connection refused
</pre>
</html>
<<newinfo "You <strong>can't</strong> connect to Redis at localhost:36379 from the command line <strong>inside the container</strong>">>
<p>
</p>
<div class="nav">
[[I know what to do -> Guess answer]]
[[Summarize what we've learned so far -> Summarize]]
</div><p> Here's what we've learned from our experiments!</p>
<p>
<strong>Inside the Python container:</strong>
</p>
<p>
Connecting to <code>localhost:36379</code> <strong> doesn't </strong> work. It
doesn't work on the command line, and it also doesn't work from inside our
Python program. So it's consistent at least :)
</p>
<p>
<strong>Outside the container: (from the command line inside your normal terminal)</strong>:
</p>
<p>
Connecting to <code>localhost:36379</code> <strong> does</strong> work. That
tells us that Redis is running, though it doesn't explain how we're supposed to
connect to Redis from the inside the container.
</p>
<<ask "How can we connect to Redis from inside the container?" "$fix" "guess 2">><p>
Beware: this game has no way of checking if your answer is right, so it'll tell you the answer on the next page :)
</p>
<<ask "How do we fix the bug?" "$fix" "guess 2">>
<div class="nav">
[[Go back -> investigate 3]]
</div><p>
Beware: this game has no way of checking if your answer is right, so it'll tell you the answer on the next page :)
</p>
<<ask "How do we fix the bug?" "$fix" "guess 2">>
<div class="nav">
[[Go back -> Error]]
</div><<you-said "You said:" $fix>>
<p>
This bug is happening because we're using the wrong hostname and port for Redis.
</p>
<p>
The address we need to use is <code>redis:6379</code>. This is because the
Python container is <strong>inside</strong> a container, and Docker Compose has
set up a DNS server so that <code>redis</code> refers to the Redis container's
IP address.
</p>
<p>
You check to see that <code>redis:6379</code> actually works:
</p>
<html>
<pre>
$ docker-compose exec backend /bin/bash <br>
root@bbcf2157926e:/# redis-cli -h redis -p 6379 keys "*" <br>
(empty list or set)
</pre>
</html>
<p>
Hooray! It works!
</p>
<div class="nav">
[[Fix the Python program -> fix Python]]
</div>You edit the Python code to replace <code>localhost</code> with <code>redis</code> and <code>36379</code> with <code>6379</code>:
<html>
<pre>
import redis <br>
import flask <br>
<br>
app = flask.Flask(__name__) <br>
r = redis.Redis(host='redis', port=6379, db=0) <br>
<br>
@app.route('/count') <br>
def count(): <br>
r.incr('hits') <br>
return 'This page has been viewed %s times.' % r.get('hits') <br>
<br>
app.run(host='0.0.0.0', port=3000)
</pre>
</html>
<p>
(side note: in real life you would probably use an environment variable for the Redis host and port instead of hardcoding them, but we're keeping it simple here :)
</p>
<div class="nav">
[[Check if it works -> you win]]
</div><p>
The bug is gone! Redis is working perfectly! Hooray!
</p>
<div style="text-align:center">
<img style="display: inline-block; max-width: 400px" src=" ">
</div>
<div class="nav">
[[Epilogue->Epilogue]]
</div><p>
This is a very common container networking problem with Docker Compose. It's
counterintuitive because you might be used to mapping container ports to ports
on the host, but from <i>another</i> container, the host port won't work.
</p>
<p>
This feature where you can use the name of the container (<code>redis</code>)
to connect from another container is specific to Docker Compose -- it doesn't
work in regular Docker
</p>
Let's look one last time at the different results trying to connect to Redis from inside/outside the container:
<html>
<pre>
$ redis-cli -h localhost -p 36379 keys "*" <br>
(empty array) <br>
$ redis-cli -h redis -p 6379 keys "*" <br>
Could not connect to Redis at redis:6379: Name or service not known <br>
$ docker-compose exec backend /bin/bash <br>
root@bbcf2157926e:/# redis-cli -h localhost -p 36379 keys "*" <br>
Could not connect to Redis at localhost:36379: Connection refused <br>
root@bbcf2157926e:/# redis-cli -h redis -p 6379 keys "*" <br>
(empty list or set)
</pre>
</html>
<p>
This bug is very simple to fix once you know about it, but it can be pretty
confusing the first time you run into it.
</p>Here are the contents of <code>docker-compose.yaml</code>. This is the file
where the container's networking is configured.
<html>
<pre>
version: "3" <br>
services: <br>
backend: <br>
build: '.' <br>
ports: <br>
- "3000:3000" <br>
redis: <br>
image: redis <br>
container_name: cache <br>
ports: <br>
- 36379:6379 <br>
restart: always
</pre>
</html>
<p>
Does this look right to you?
</p>
<div class="nav">
[[Yes, go back -> Python program]]
[[I'm not sure -> docker-compose 2]]
</div><p>
Everything looks good! This file is mapping port 6379 inside the <code>redis</code> container to port 36379 outside the
container, and port 3000 inside the <code>backend</code> container to port 3000 outside the container.
</p>
<p>
One fact that's not obvious here is that in a container in Docker Compose, you
can use the <code>redis</code> hostname to connect to the Redis container. For
example, if you're in the <code>backend</code> container,
<code>redis:6379</code> will connect to the Redis server.
</p>
[[Go back -> Python program]]You try to ping the container to see if it's up, but can't figure out how to do
it. After some quick Googling, it looks like you can't ping a Docker container
on Mac.
<div class="nav">
[[Go back -> Python program]]
</div>