Angular-CLI代理在Docker中无法工作

4
我有一个包含我项目所需的所有内容的docker-compose文件。
这个docker-compose包含nginx服务器、mysql、phpmyadmin和php。最近我添加了一个angular容器。如果我访问localhost:4200,一切正常,我可以使用angular应用程序;如果我访问localhost:80,我就可以使用Laravel后端。
现在我需要向后端API发送一个简单的请求。我为angular设置了代理,代码如下:
{
   "/api/*": {
      "target":"http://localhost:80",
      "secure":false,
      "changeOrigin":true,
      "pathRewrite": {"^/api" : ""},
      "logLevel":"debug"
   }
}

这是我基于这篇文章复制的配置。但是当我尝试调用时,Chrome说http://localhost:4200/api/test不存在(错误404),这是正常的。另一方面,Angular服务器显示
HPM]在尝试将来自localhost:4200的请求代理到http://localhost:80时发生错误(ECONNREFUSED)

我猜这是由docker引起的,但我不知道如何解决。

[编辑]

version: '2'

services:
  web_server:
    restart: always
    image: nginx:latest
    ports:
      - "80:80"
    volumes:
      - /projects/laravel/:/var/www/                                               
      - /docker/sites-enabled-nginx:/etc/nginx/sites-enabled/     
      - /docker/nginx.conf:/etc/nginx/nginx.conf
    links:
      - php:php

  php:
    restart: always
    build: /docker/php/
    container_name: "php"
    volumes:
      - /projects/laravel/:/var/www/       
      - db:db

  db:
    restart: always
    image: mysql
    volumes:
      - /Users/Irindul/mysql:/var/lib/mysql     
    ports:
      - "3306:3306"                                   
    container_name: "mysql"
    environment:
      - MYSQL_DATABASE=homestead
      - MYSQL_USER=homestead
      - MYSQL_PASSWORD=secret
      - MYSQL_ROOT_PASSWORD=root

  myadmin:
    restart: always
    image: phpmyadmin/phpmyadmin
    links:
      - db:db
    ports:
      - "8080:80"                                     



  angular:
    restart: always
    build: /docker/angular
    container_name: angular
    volumes:
      - /projects/angular/package.json:/usr/src/app/package.json
      - /projects/angular:/usr/src/app/
    ports:
      - "4200:4200"

以下是PHP和Angular的Dockerfiles:

PHP:

FROM php:7-fpm
RUN docker-php-ext-install pdo pdo_mysql
WORKDIR /var/www

Angular :

#Latest Node
FROM node

#Creating working folder
RUN mkdir -p /usr/src/app

#Update pwd
WORKDIR /usr/src/app

#Run npm
RUN npm install
EXPOSE 4200
CMD ["npm", "start"]

package.json :

{
  "name": "client",
  "version": "0.0.0",
  "license": "MIT",
  "scripts": {
    "ng": "ng",
    "start": "ng serve -H 0.0.0.0 --proxy-config proxy.config.json",
    "build": "ng build",
    "test": "ng test",
    "lint": "ng lint",
    "e2e": "ng e2e"
  },
  "private": true,
  "dependencies": {
    "@angular/animations": "^4.2.4",
    "@angular/common": "^4.2.4",
    "@angular/compiler": "^4.2.4",
    "@angular/core": "^4.2.4",
    "@angular/forms": "^4.2.4",
    "@angular/http": "^4.2.4",
    "@angular/platform-browser": "^4.2.4",
    "@angular/platform-browser-dynamic": "^4.2.4",
    "@angular/router": "^4.2.4",
    "core-js": "^2.4.1",
    "rxjs": "^5.4.2",
    "zone.js": "^0.8.14"
  },
  "devDependencies": {
    "@angular/cli": "1.3.2",
    "@angular/compiler-cli": "^4.2.4",
    "@angular/language-service": "^4.2.4",
    "@types/jasmine": "~2.5.53",
    "@types/jasminewd2": "~2.0.2",
    "@types/node": "~6.0.60",
    "codelyzer": "~3.1.1",
    "jasmine-core": "~2.6.2",
    "jasmine-spec-reporter": "~4.1.0",
    "karma": "~1.7.0",
    "karma-chrome-launcher": "~2.1.1",
    "karma-cli": "~1.0.1",
    "karma-coverage-istanbul-reporter": "^1.2.1",
    "karma-jasmine": "~1.1.0",
    "karma-jasmine-html-reporter": "^0.2.2",
    "protractor": "~5.1.2",
    "ts-node": "~3.2.0",
    "tslint": "~5.3.2",
    "typescript": "~2.3.3"
  }
}

添加Compose文件,这是必需的。 - Tarun Lalwani
@TarunLalwani 我编辑了我的帖子并添加了Compose文件。 - Irindul
请添加你的 package.json 文件,我没有看到你在运行时进行 npm 安装?你也需要在运行时挂载 package.json,并运行 npm install。 - Tarun Lalwani
如果你查看你的Dockerfile,当你执行npm install时,容器中没有package.json吗?你根本没有复制它。 - Tarun Lalwani
好的,我最初确实复制了它,但由于我将其挂载在docker-compose中,所以我认为这并不是必要的。 - Irindul
显示剩余2条评论
5个回答

5
为了解决这个问题,我在docker-compose文件中的前端/客户端服务中创建了一个链接,用于连接我的API服务:

客户端

  # Frontend service
  client:
    build: client # specify the directory of the Dockerfile
    ports:
      - "4200:4200" # specify port forewarding
    links: 
      - api # link this service to the api service

API

  # API Service
  api: 
    build: api # specify the directory of the Dockerfile
    ports:
      - "5000:5000" #specify ports forewarding
    links:
      - database # link this service to the database service

接着,我在angular/client文件夹中更新了我的代理配置文件:

{
  "/api/*": {
    "target": "http://api:5000/",
    "secure": false,
    "changeOrigin": true
  }
}

注意: 在docker composer文件中,目标URL是相同的: api,不要忘记暴露的端口,在我的情况下是5000。


1
链接不再建议使用,因为它是一个已弃用的功能。还有其他建议吗? - Vytautas Arminas

3

经过大量调试,我终于使它正常工作了。

@MSLacerda的解决方案可以运行,但我想添加一个使用docker网络的答案,因为docker连接已经被弃用。

docker-compose.yaml

version: '3'

services: 
  client:
    # all other configurations...
    ports:
      - 4200:4200
    networks:
      - backend-net
  api:
    # all other configurations...
    networks:
      - backend-net

networks:
  backend-net:
    driver: bridge

proxy.config.json

{
  "/api/*": {
    "target": "http://api:3000",
    "secure": false,
    "changeOrigin": true
    "pathRewrite": {
      "^/api": ""
    }
  }
}

需要注意的是,请确保proxy.config.json中的目标主机是您想要连接的服务名称。(在这种情况下,您希望客户端连接到api服务)。Docker将动态地将主机替换为容器的IP地址。别忘了为目标添加端口。


1
您的问题在于没有复制package.json文件,当没有package.json文件时,npm install命令不会执行任何操作。
请将您的angular文件更改为以下内容:
#Latest Node
FROM node

#Creating working folder
RUN mkdir -p /usr/src/app -p /usr/src/node_modules
ENV NODE_PATH=/usr/src/node_modules

WORKDIR /usr/src/app
COPY package.json ./package.json
RUN npm install
EXPOSE 4200
CMD ["npm", "start"]

这将在$NODE_PATH中安装您的node_nodules,该路径为/usr/src/node_modules
将您的Angular服务在compose中更改如下。
 angular:
    restart: always
    build: /docker/angular
    container_name: angular
    volumes:
      - /projects/angular:/usr/src/app/
    ports:
      - "4200:4200"

不需要为package.json添加额外条目。即使您从本地文件夹挂载覆盖了/usr/src/app/node_modules,但我们已将节点模块更改为在/usr/src/node_modules查找。

仍然没有解决我的代理问题,我收到了相同的错误。 - Irindul

0

实际上,您想要做的是设置一个“反向代理”。而在 docker 容器内部,这不是正确的方法。

您需要在容器中的 nginx 文件夹下的“default.conf”文件中添加代理。

  1. 使用以下命令将容器中的 default.conf 文件复制到本地机器

docker cp [您的容器 ID]:/etc/nginx/conf.d/default.conf [您的本地机器目标地址]

  1. 编辑 default.conf 文件并像这样添加代理:
location /api{
   proxy_pass http://newURL;
}
  1. 使用以下命令将修改后的文件复制回您的Docker容器

docker cp [您本地机器的目标地址] [您的容器ID]:/etc/nginx/conf.d/default.conf

  1. 重新启动您的Docker容器。

-14

我通过将 Angular 从 Docker 中移除并手动运行简单的 npm start 命令来解决了问题。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接