从此
📄文章 #️⃣专题 🌐酷站 👨‍💻技术 📺 📱

Nginx反向代理、HAProxy负载均衡、Java HTTP Client、Jetty Web Server、NanoHTTPD Web Server

综合/最新



Nginx

综合:
  命令:
    禁用开机启动 sudo systemctl disable nginx

  巡查:
    访问日志  sudo cat /var/log/nginx/access.log
    报错日志  sudo cat /var/log/nginx/error.log

  会走HTTP/1.X头Upgrade至HTTP/2的ALPN协商流程: curl --http2 localhost:8080
    说明 - Connection头此处是为了告诉中间代理,Upgrade头仅用于首跳使用,不要原样向上游传。
    Connection: Upgrade
    Upgrade: h2

  不走HTTP/1.X头Upgrade至HTTP/2的ALPN协商流程(无需配置证书): curl --http2-prior-knowledge localhost:8080
    server {
	listen 8080 http2;
	location / {
		return 200 "it's ok!";
	}
    }

http跳转https:
      server {
        listen 80 default_server; server_name _;
        
        return 301 https://$host$request_uri;
      }

反向代理:
  proxy_pass 缓存状态取决于目标网页的响应头配置,或自定义proxy_cache。

  最简配置:
    location / {
      
      proxy_pass http://localhost:8000;
      
      proxy_set_header Host $host;
    }

  最佳实践:
    location / {
      
      proxy_pass http://localhost:8000;
      
      proxy_set_header Host $host;

            
      proxy_http_version 1.1;
      
      proxy_set_header Connection "";

      
      proxy_set_header X-Forwarded-Proto $scheme;
            
      proxy_set_header X-Real-IP $remote_addr;
    }

    location = /manifest.json {
      default_type application/json;
      return 200 '{"name": "应用名字", "id": "唯一标识"}';
    }

上传文件Size限制:
    location ^~ /main/apis/more/default/public/upload-to-storage {
      try_files $uri @backend;
      client_max_body_size 10m;  # 暂用前端限制,Nginx只设最大值。
    }

CORS跨域:
    location / {
        # 允许所有域名跨域
        add_header 'Access-Control-Allow-Origin' '*';
        add_header 'Access-Control-Allow-Methods' 'GET, POST, OPTIONS';
        add_header 'Access-Control-Allow-Headers' 'DNT,User-Agent,X-Requested-With,If-Modified-Since,Cache-Control,Content-Type,Range';
        add_header 'Access-Control-Expose-Headers' 'Content-Length,Content-Range';

        # 处理 OPTIONS 预检请求;缓存时效为2小时。
        if ($request_method = 'OPTIONS') {
            add_header 'Access-Control-Max-Age' 7200;
            add_header 'Content-Length' 0;
            return 204;
        }

        # 其他配置
        proxy_pass http://backend;
        proxy_set_header Host $host;
    }

SSE流式通讯:
  浏览器以listen段配置的HTTP/2协议为准,且SSE从HTTP/1.1起就支持了,且 proxy_pass 对SSE的连接数未做限制,故影响不到浏览器的连接数限制(6个/HTTP2默认100个)。
    location /sse {
      proxy_pass http://ip:9080; #  SSE不支持try_files指令
      proxy_set_header Host $host;

      proxy_http_version 1.1;
      proxy_set_header Connection "";
      proxy_set_header X-Forwarded-Proto $scheme;
      proxy_set_header X-Real-IP $remote_addr;

      proxy_buffering off; // 或业务端控制 response.setHeader("X-Accel-Buffering", "no");
      proxy_read_timeout 12h; # Nginx默认1分钟。
    }

WebSocket配置:
  location ^~ /websocket/ {
      proxy_pass http://ip:9080; #  支持try_files指令?
      proxy_set_header Host $host;

      proxy_http_version 1.1;
      #proxy_set_header Connection "";
      proxy_set_header X-Forwarded-Proto $scheme;
      proxy_set_header X-Real-IP $remote_addr;

      proxy_set_header Upgrade $http_upgrade;
      proxy_set_header Connection "upgrade";
      #proxy_read_timeout 12h; # Nginx默认1分钟;要么加长超时,要么上WebSocket Ping Pong机制。
  }

HTTP Web C/S

Java HTTP Client
SSE客户端则为jakarta.ws.rs.sse.SseEventSource,或java.net.http.HttpClient响应至超时,若存在结束标记,则可调HttpClient.shutdownNow()避免一直等到超时。
  implementation("org.glassfish.jersey.media:jersey-media-sse:4.0.0-M2")
  implementation("org.apache.cxf:cxf-rt-rs-sse:4.1.0")

H2C:
  Netty需要显式设置HttpClient.create().protocol(HttpProtocol.H2C); 而 java.net.http.HttpClient 则不需要。
        var bv=Base64.getEncoder().encodeToString("clientId:clientSecret".getBytes());
        var r = HttpClient.newHttpClient().send(HttpRequest.newBuilder(
                URI.create("https://example.com/oauth2/token"))
                .POST(HttpRequest.BodyPublishers.ofString("grant_type=refresh_token&refresh_token=" + rt))
                .header("Content-Type", "application/x-www-form-urlencoded")
                .header("Authorization", "Basic " + bv) // or JS - btoa("clientId:clientSecret")
                .build(), HttpResponse.BodyHandlers.ofString()).body();
        System.out.println(r);

  POST multipart/form-data数据:
    // implementation("org.apache.httpcomponents.client5:httpclient5:5.4.2")
    var me = MultipartEntityBuilder.create().addTextBody("substringLength", "36")
            //.addTextBody("dataText", "x", ContentType.TEXT_PLAIN.withCharset(Charset.defaultCharset()))
            .build();
    var baos = new ByteArrayOutputStream();
    me.writeTo(baos);

    var r = HttpClient.newHttpClient().send(HttpRequest.newBuilder(
            java.net.URI.create("https://localhost/path/url-slug"))
            .POST(HttpRequest.BodyPublishers.ofByteArray(baos.toByteArray()))
            //ContentType.MULTIPART_FORM_DATA.withParameters(new BasicNameValuePair("boundary", ctb)).toString()
            .header("Content-Type", me.getContentType())
            .build(), HttpResponse.BodyHandlers.ofString()).body();
    System.out.println(r);
Jetty Servlet Server
Jetty反向代理: org.eclipse.jetty.ee10.proxy.AsyncProxyServlet兼容javax或jakarta的Servlet代理库

解决 ResourceHandler + setDirAllowed(true) 报 “Ambiguous URI empty segment”:
  server.getConnectors()[0].getConnectionFactory(HttpConnectionFactory.class).getHttpConfiguration().setUriCompliance(UriCompliance.LEGACY);
  或 new HttpConfiguration().setUriCompliance(UriCompliance.LEGACY);

Jetty最精简HttpServer: 
    // implementation("org.eclipse.jetty:jetty-server:12.0.16")
    public static void main(String[] args) throws Exception {
        var server = new Server(); // Android 14 及早期版本均报 Didn't find class "java.util.ServiceLoader$Provider",可换用 jetty-server:9.x 或 NanoHTTPD;Android 15 则添加了该类;desugar_jdk_libs 补全不了。
        var sc = new ServerConnector(server);
        sc.setPort(8080); // 入参0则端口由系统分配,取端口号:int port = ((ServerConnector) ss.getConnectors()[0]).getLocalPort();
        server.addConnector(sc);
        server.setHandler(new Handler.Abstract() {
            @Override
            public boolean handle(Request request, Response response, Callback callback) {
                System.out.println("handle complete - http://localhost:8080/");
                callback.succeeded();
                return true;
            }
        });
        /* 
          // Servlet - implementation("org.eclipse.jetty.ee10:jetty-ee10-servlet:12.0.16")
          var context = new ServletContextHandler();
          context.addServlet(staticServlet.class, "/");
          server.setHandler(context);

          var rh = new ResourceHandler();
          rh.setBaseResourceAsString(tmpDir);
          rh.setDirAllowed(true);
          // 目录别斜杠结尾,否则会变双斜杠。
          server.setHandler(new ContextHandler(rh, "/.well-known/tmp"));
        */
        server.start();
    }
    public static class staticServlet extends HttpServlet {

        @Override
        protected void doGet(HttpServletRequest request,
                HttpServletResponse response) throws IOException {
            response.getWriter().println("x");
        }
    }
NanoHTTPD Web Server
    implementation("org.nanohttpd:nanohttpd-webserver:2.3.1")
    // https://github.com/NanoHttpd/nanohttpd/blob/master/webserver/src/main/java/org/nanohttpd/webserver/SimpleWebServer.java
    var server = new fi.iki.elonen.SimpleWebServer("localhost", 9458, MainActivity.this.getFilesDir(), true, "*");
    try {
        server.start(NanoHTTPD.SOCKET_READ_TIMEOUT, false);
    } catch (IOException ioe) {
        System.err.println("Couldn't start server:\n" + ioe);
        //System.exit(-1);
    }

HAProxy

说明:
  HAProxy backend 支持 h2c,而 Nginx proxy_pass 最高只支持到 HTTP/1.1;默认上传文件大小未做限制。
  frontend 配置浏览器至 HAProxy 的访问协议,backend 配置 HAProxy 反向代理的访问协议;mode 默认值为 tcp,建站则通常配置为 mode http,其值http含https。

官方文档 - https://www.haproxy.org

安装/管理:
  sudo apt -y install haproxy
  sudo cp /etc/haproxy/haproxy.cfg .
  说明 - 该配置文件未配置frontend和backend,需自行配置;default_backend 为兜底配置。
  sudo vim.tiny /etc/haproxy/haproxy.cfg

frontend f1
    mode http
    bind :80  # http to https; 可与bind :443 ...并存
    #http-request redirect scheme https unless { ssl_fc }
    #bind :443 ssl crt /etc/haproxy/our.pem alpn h2,http/1.1
    acl a-name hdr(host) -i abc.passed.app
    use_backend b1 if a-name
    default_backend b2

backend b1
    mode http
    server s1 23.192.228.84:80

backend b2
    mode http
    server s2 23.192.228.84:80

  sudo systemctl restart haproxy
  curl.exe -HHost:example.com http://34.169.255.233 -I

用法:
  隐式 and 条件:if { path_beg /images/ } { method POST } 若显式写上 and 则报:no such ACL : 'and'
  自头匹配用 path_beg;精确匹配用 path:
    http-request redirect code 301 location https://tsc.openle.com/.well-known/temporary/favicon-temporary.ico if { path /favicon.ico }

  取正则表达式匹配部分:
    http-request redirect code 301 location /?url=%[path,'regsub("^/main/apis/more/risky/mixed-content/(.*)$","\1")'] if risky-path

    # capture.req.uri指完整URL。
    http-request set-var(txn.user_id) capture.req.uri
    # field指按冒号分隔取第1个。
    http-request set-var(txn.user_id) capture.req.uri,field(1,:)
    http-request redirect code 301 location /?url=%[var(txn.user_id)] if risky-path

  H2C 配置:
    mode http
    server s1 192.168.0.10:80 proto h2

其他