我有一个在节点容器和 Redis 容器中运行的 Express API。 当尝试将节点容器连接到 Redis 时,出现以下错误。
api_1 | events.js:291
api_1 | throw er; // Unhandled 'error' event
api_1 | ^
api_1 |
api_1 | Error: Redis connection to localhost:6379 failed - connect ECONNREFUSED 127.0.0.1:6379
api_1 | at TCPConnectWrap.afterConnect [as oncomplete] (net.js:1144:16)
api_1 | Emitted 'error' event on RedisAdapter instance at:
api_1 | at RedisClient.onError (/usr/src/api/node_modules/socket.io-redis/dist/index.js:65:22)
api_1 | at RedisClient.emit (events.js:314:20)
api_1 | at RedisClient.on_error (/usr/src/api/node_modules/redis/index.js:342:14)
api_1 | at Socket.<anonymous> (/usr/src/api/node_modules/redis/index.js:223:14)
api_1 | at Socket.emit (events.js:314:20)
api_1 | at emitErrorNT (internal/streams/destroy.js:92:8)
api_1 | at emitErrorAndCloseNT (internal/streams/destroy.js:60:3)
api_1 | at processTicksAndRejections (internal/process/task_queues.js:84:21) {
api_1 | errno: 'ECONNREFUSED',
api_1 | code: 'ECONNREFUSED',
api_1 | syscall: 'connect',
api_1 | address: '127.0.0.1',
api_1 | port: 6379
api_1 | }
这里是建立连接的代码:
const redisClient = require("redis").createClient({host: process.env.REDIS_HOST ? process.env.REDIS_HOST : 'localhost'});
我尝试将“localhost”更改为“redis”,再更改为“redis://redis:6379”。没有任何效果。
据我了解,容器上下文中的“localhost”指的是其自己的内部网络,但是nodejs和redis都在我在docker compose文件中指定的同一网络上运行。
Docker 撰写片段
api:
build: ./api
ports:
- "3004:3004"
volumes:
- type: bind
source: ./api
target: /usr/src/api
working_dir: /usr/src/api
depends_on:
- redis
networks:
- backend
redis:
image: "redis:latest"
ports:
- "6379:6379"
hostname: redis
networks:
- backend
```
What could this be. All the "solutions" that I've managed to find through research don't work for me.
为了结束这个问题,当尝试连接到容器内的 localhost 时,它假定容器本身,并且因为 node.js 容器没有在 redis 端口上发布,所以连接被拒绝。另一方面,因为使用 docker-compose,您会获得将它们绑定在一起的默认 docker 网络,因此可以使用容器\服务名称作为内部安全 DNS 记录。
有关 docker 网络内部工作的更详细说明,请参阅 此线程
确保您使用
127.0.0.1
而不是 localhost
连接到 Redis 或任何其他服务。在某些时候,Redis(和其他)连接到服务器的逻辑发生了变化。如果您使用 localhost
进行连接,则来自另一个答案:“MySQL 客户端[将]尝试使用 unix 套接字进行连接。”