All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.whaleal.icefrog.http.server.HttpServerRequest Maven / Gradle / Ivy

The newest version!
package com.whaleal.icefrog.http.server;

import com.sun.net.httpserver.Headers;
import com.sun.net.httpserver.HttpExchange;
import com.whaleal.icefrog.core.collection.CollUtil;
import com.whaleal.icefrog.core.io.IORuntimeException;
import com.whaleal.icefrog.core.io.IoUtil;
import com.whaleal.icefrog.core.map.CaseInsensitiveMap;
import com.whaleal.icefrog.core.map.multi.ListValueMap;
import com.whaleal.icefrog.core.net.NetUtil;
import com.whaleal.icefrog.core.net.multipart.MultipartFormData;
import com.whaleal.icefrog.core.net.multipart.UploadSetting;
import com.whaleal.icefrog.core.util.ArrayUtil;
import com.whaleal.icefrog.core.util.CharsetUtil;
import com.whaleal.icefrog.core.util.StrUtil;
import com.whaleal.icefrog.http.Header;
import com.whaleal.icefrog.http.HttpUtil;
import com.whaleal.icefrog.http.Method;
import com.whaleal.icefrog.http.useragent.UserAgent;
import com.whaleal.icefrog.http.useragent.UserAgentUtil;

import java.io.IOException;
import java.io.InputStream;
import java.net.HttpCookie;
import java.net.URI;
import java.nio.charset.Charset;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Map;

/**
 * Http请求对象,对{@link HttpExchange}封装
 *
 * @author Looly
 * @author wh
 * @since 1.0.0
 */
public class HttpServerRequest extends HttpServerBase {

    private Map cookieCache;
    private ListValueMap paramsCache;
    private MultipartFormData multipartFormDataCache;
    private Charset charsetCache;
    private byte[] bodyCache;

    /**
     * 构造
     *
     * @param httpExchange {@link HttpExchange}
     */
    public HttpServerRequest( HttpExchange httpExchange ) {
        super(httpExchange);
    }

    /**
     * 获得Http Method
     *
     * @return Http Method
     */
    public String getMethod() {
        return this.httpExchange.getRequestMethod();
    }

    /**
     * 是否为GET请求
     *
     * @return 是否为GET请求
     */
    public boolean isGetMethod() {
        return Method.GET.name().equalsIgnoreCase(getMethod());
    }

    /**
     * 是否为POST请求
     *
     * @return 是否为POST请求
     */
    public boolean isPostMethod() {
        return Method.POST.name().equalsIgnoreCase(getMethod());
    }

    /**
     * 获得请求URI
     *
     * @return 请求URI
     */
    public URI getURI() {
        return this.httpExchange.getRequestURI();
    }

    /**
     * 获得请求路径Path
     *
     * @return 请求路径
     */
    public String getPath() {
        return getURI().getPath();
    }

    /**
     * 获取请求参数
     *
     * @return 参数字符串
     */
    public String getQuery() {
        return getURI().getQuery();
    }

    /**
     * 获得请求header中的信息
     *
     * @return header值
     */
    public Headers getHeaders() {
        return this.httpExchange.getRequestHeaders();
    }

    /**
     * 获得请求header中的信息
     *
     * @param headerKey 头信息的KEY
     * @return header值
     */
    public String getHeader( Header headerKey ) {
        return getHeader(headerKey.toString());
    }

    /**
     * 获得请求header中的信息
     *
     * @param headerKey 头信息的KEY
     * @return header值
     */
    public String getHeader( String headerKey ) {
        return getHeaders().getFirst(headerKey);
    }

    /**
     * 获得请求header中的信息
     *
     * @param headerKey 头信息的KEY
     * @param charset   字符集
     * @return header值
     */
    public String getHeader( String headerKey, Charset charset ) {
        final String header = getHeader(headerKey);
        if (null != header) {
            return CharsetUtil.convert(header, CharsetUtil.CHARSET_ISO_8859_1, charset);
        }
        return null;
    }

    /**
     * 获取Content-Type头信息
     *
     * @return Content-Type头信息
     */
    public String getContentType() {
        return getHeader(Header.CONTENT_TYPE);
    }

    /**
     * 获取编码,获取失败默认使用UTF-8,获取规则如下:
     *
     * 
     *     1、从Content-Type头中获取编码,类似于:text/html;charset=utf-8
     * 
* * @return 编码,默认UTF-8 */ public Charset getCharset() { if (null == this.charsetCache) { final String contentType = getContentType(); final String charsetStr = HttpUtil.getCharset(contentType); this.charsetCache = CharsetUtil.parse(charsetStr, DEFAULT_CHARSET); } return this.charsetCache; } /** * 获得User-Agent * * @return User-Agent字符串 */ public String getUserAgentStr() { return getHeader(Header.USER_AGENT); } /** * 获得User-Agent,未识别返回null * * @return User-Agent字符串,未识别返回null */ public UserAgent getUserAgent() { return UserAgentUtil.parse(getUserAgentStr()); } /** * 获得Cookie信息字符串 * * @return cookie字符串 */ public String getCookiesStr() { return getHeader(Header.COOKIE); } /** * 获得Cookie信息列表 * * @return Cookie信息列表 */ public Collection getCookies() { return getCookieMap().values(); } /** * 获得Cookie信息Map,键为Cookie名,值为HttpCookie对象 * * @return Cookie信息Map */ public Map getCookieMap() { if (null == this.cookieCache) { cookieCache = Collections.unmodifiableMap(CollUtil.toMap( NetUtil.parseCookies(getCookiesStr()), new CaseInsensitiveMap<>(), HttpCookie::getName)); } return cookieCache; } /** * 获得指定Cookie名对应的HttpCookie对象 * * @param cookieName Cookie名 * @return HttpCookie对象 */ public HttpCookie getCookie( String cookieName ) { return getCookieMap().get(cookieName); } /** * 是否为Multipart类型表单,此类型表单用于文件上传 * * @return 是否为Multipart类型表单,此类型表单用于文件上传 */ public boolean isMultipart() { if (false == isPostMethod()) { return false; } final String contentType = getContentType(); if (StrUtil.isBlank(contentType)) { return false; } return contentType.toLowerCase().startsWith("multipart/"); } /** * 获取请求体文本,可以是form表单、json、xml等任意内容
* 使用{@link #getCharset()}判断编码,判断失败使用UTF-8编码 * * @return 请求 */ public String getBody() { return getBody(getCharset()); } /** * 获取请求体文本,可以是form表单、json、xml等任意内容 * * @param charset 编码 * @return 请求 */ public String getBody( Charset charset ) { return StrUtil.str(getBodyBytes(), charset); } /** * 获取body的bytes数组 * * @return body的bytes数组 */ public byte[] getBodyBytes() { if (null == this.bodyCache) { this.bodyCache = IoUtil.readBytes(getBodyStream(), true); } return this.bodyCache; } /** * 获取请求体的流,流中可以读取请求内容,包括请求表单数据或文件上传数据 * * @return 流 */ public InputStream getBodyStream() { return this.httpExchange.getRequestBody(); } /** * 获取指定名称的参数值,取第一个值 * * @param name 参数名 * @return 参数值 * @since 1.0.0 */ public String getParam( String name ) { return getParams().get(name, 0); } /** * 获取指定名称的参数值 * * @param name 参数名 * @return 参数值 * @since 1.0.0 */ public List getParams( String name ) { return getParams().get(name); } /** * 获取参数Map * * @return 参数map */ public ListValueMap getParams() { if (null == this.paramsCache) { this.paramsCache = new ListValueMap<>(); final Charset charset = getCharset(); //解析URL中的参数 final String query = getQuery(); if (StrUtil.isNotBlank(query)) { this.paramsCache.putAll(HttpUtil.decodeParams(query, charset)); } // 解析multipart中的参数 if (isMultipart()) { this.paramsCache.putAll(getMultipart().getParamListMap()); } else { // 解析body中的参数 final String body = getBody(); if (StrUtil.isNotBlank(body)) { this.paramsCache.putAll(HttpUtil.decodeParams(body, charset)); } } } return this.paramsCache; } /** * 获取客户端IP * *

* 默认检测的Header: * *

     * 1、X-Forwarded-For
     * 2、X-Real-IP
     * 3、Proxy-Client-IP
     * 4、WL-Proxy-Client-IP
     * 
* *

* otherHeaderNames参数用于自定义检测的Header
* 需要注意的是,使用此方法获取的客户IP地址必须在Http服务器(例如Nginx)中配置头信息,否则容易造成IP伪造。 *

* * @param otherHeaderNames 其他自定义头文件,通常在Http服务器(例如Nginx)中配置 * @return IP地址 */ public String getClientIP( String... otherHeaderNames ) { String[] headers = {"X-Forwarded-For", "X-Real-IP", "Proxy-Client-IP", "WL-Proxy-Client-IP", "HTTP_CLIENT_IP", "HTTP_X_FORWARDED_FOR"}; if (ArrayUtil.isNotEmpty(otherHeaderNames)) { headers = ArrayUtil.addAll(headers, otherHeaderNames); } return getClientIPByHeader(headers); } /** * 获取客户端IP * *

* headerNames参数用于自定义检测的Header
* 需要注意的是,使用此方法获取的客户IP地址必须在Http服务器(例如Nginx)中配置头信息,否则容易造成IP伪造。 *

* * @param headerNames 自定义头,通常在Http服务器(例如Nginx)中配置 * @return IP地址 * @since 1.0.0 */ public String getClientIPByHeader( String... headerNames ) { String ip; for (String header : headerNames) { ip = getHeader(header); if (false == NetUtil.isUnknown(ip)) { return NetUtil.getMultistageReverseProxyIp(ip); } } ip = this.httpExchange.getRemoteAddress().getHostName(); return NetUtil.getMultistageReverseProxyIp(ip); } /** * 获得MultiPart表单内容,多用于获得上传的文件 * * @return MultipartFormData * @throws IORuntimeException IO异常 * @since 1.0.0 */ public MultipartFormData getMultipart() throws IORuntimeException { if (null == this.multipartFormDataCache) { this.multipartFormDataCache = parseMultipart(new UploadSetting()); } return this.multipartFormDataCache; } /** * 获得multipart/form-data 表单内容
* 包括文件和普通表单数据
* 在同一次请求中,此方法只能被执行一次! * * @param uploadSetting 上传文件的设定,包括最大文件大小、保存在内存的边界大小、临时目录、扩展名限定等 * @return MultiPart表单 * @throws IORuntimeException IO异常 * @since 1.0.0 */ public MultipartFormData parseMultipart( UploadSetting uploadSetting ) throws IORuntimeException { final MultipartFormData formData = new MultipartFormData(uploadSetting); try { formData.parseRequestStream(getBodyStream(), getCharset()); } catch (IOException e) { throw new IORuntimeException(e); } return formData; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy