背景
因为业务上的需求,需要将数据传递给第三方,第三方以表单形式接收包含数据的文件,而我方数据存在数据库中,按常规做法,先把数据保存在本地文件,在利用http工具类上传,因为数据行数过多,文件io会大大影响效率,所以决定跳过文件io。
方案
因为HTTP属于应用层协议,所以可以在代码中拼装上传文件的HTTP报文,通过scoket打开TCP连接,写入报文。
过程
首先通过Wireshark抓取环回地址文件上传的HTTP报文:
然后参照图片中报文的样子拼装http
bufferedWriter.write("POST " + path + " HTTP/1.1\r\n");
bufferedWriter.write("Host: " + "127.0.0.1:8080" + "\r\n");
bufferedWriter.write("content-type: multipart/form-data; boundary=--sugar963sugar\r\n");
bufferedWriter.write("content-length: 10000\r\n");
bufferedWriter.write("\r\n");
bufferedWriter.write("----sugar963sugar\r\n");
bufferedWriter.write("Content-Disposition: form-data; name=\"file\";filename=\"wx.txt\"\r\n");
bufferedWriter.write("Content-type: text/plain\r\n\r\n");
bufferedWriter.write("sdadadadadadada456555cuilvjjf645");
bufferedWriter.write("\r\n----sugar963sugar--\r\n")
拼装好后利用socket请求获取服务器成功响应
但是这个报文有个问题,必须在上传前指定数据长度:content-length: 10000\r\n
当然,可以将这个长度设置成足够业务使用的长度,但为了能够比较通用,避免硬编码,需要换一种传输方式
通过Wireshark抓包发现,http有另外一种动态传输数据的方式:
继续照葫芦画瓢:
bufferedWriter.write("POST " + path + " HTTP/1.1\r\n");
bufferedWriter.write("Host: " + "127.0.0.1:8080" + "\r\n");
bufferedWriter.write("content-type: multipart/form-data; boundary=--sugar963sugar\r\n");
bufferedWriter.write("Transfer-Encoding: chunked\r\n");
bufferedWriter.write("\r\n");
bufferedWriter.write("6c\r\n");
bufferedWriter.write("----sugar963sugar\r\n");
bufferedWriter.write("Content-Disposition: form-data; name=\"fi\";filename=\"wx.txt\"\r\n");
bufferedWriter.write("Content-type: text/plain\r\n\r\n\r\n");
bufferedWriter.write("35\r\n");
bufferedWriter.write("sdadadadadadada456555cuilvjjf645");
bufferedWriter.write("\r\n----sugar963sugar--\r\n");
bufferedWriter.write("0\r\n\r\n");
关键的头部信息:Transfer-Encoding: chunked 分块传输
利用socket发送依旧成功获得响应
自此开始编写工具类
结果
package com.example.demo.controller;
import javax.net.ssl.SSLSocketFactory;
import java.io.*;
import java.net.Socket;
import java.util.Arrays;
/**
*
* @author sugar
*
* Created by 2019/01/24
*
*
* */
public class HttpStream {
private Socket socket;
private InputStream inputStream = null;
private OutputStream outputStream = null;
DataMeta dataMeta = DataMeta.FORM;
RequestMothed requestMothed = RequestMothed.POST;
public class Response {
int status;
String msg;
}
public enum RequestMothed {
POST(1);
int value;
RequestMothed(int value) {
this.value = value;
}
}
public enum DataMeta {
FORM(1);
int value;
DataMeta(int value) {
this.value = value;
}
}
public HttpStream(RequestMothed requestMothed, DataMeta dataMeta, String host, int port, boolean isSSL, String path) {
this.dataMeta = dataMeta;
this.requestMothed = requestMothed;
try {
if (isSSL) {
//https请求
socket = (SSLSocketFactory.getDefault()).createSocket(host, port);
inputStream = socket.getInputStream();
outputStream = socket.getOutputStream();
} else {
socket = new Socket(host, port);
inputStream = socket.getInputStream();
outputStream = socket.getOutputStream();
}
if (RequestMothed.POST == requestMothed) {
outputStream.write(("POST " + path + " HTTP/1.1\r\n").getBytes());
outputStream.write(("Host: " + host + ":" + port + "\r\n").getBytes());
outputStream.write(("Connection: keep-alive\r\n").getBytes());
if (DataMeta.FORM == dataMeta) {
outputStream.write(("content-type: multipart/form-data; boundary=--sugar963sugar\r\n").getBytes());
outputStream.write(("Transfer-Encoding: chunked\r\n").getBytes());
outputStream.write(("\r\n").getBytes());
} else {
//其他数据提交方式
}
} else {
//其他请求方式
}
} catch (IOException e) {
throw new RuntimeException(e);
}
}
public void putData(String key, String value) {
try {
if (RequestMothed.POST == requestMothed) {
if (DataMeta.FORM == dataMeta) {
putDataOfForm(key, value);
} else {
//其他数据提交方式
}
} else {
//其他请求方式
}
} catch (IOException e) {
throw new RuntimeException(e);
}
}
public void putDataWithFile(String key, String fileName) {
try {
if (RequestMothed.POST == requestMothed) {
if (DataMeta.FORM == dataMeta) {
putDataOfFormWithFile(key, fileName);
} else {
//其他数据提交方式
}
} else {
//其他请求方式
}
} catch (IOException e) {
throw new RuntimeException(e);
}
}
public void putFileContent(byte[] content, boolean isEnd) {
try {
if (RequestMothed.POST == requestMothed) {
if (DataMeta.FORM == dataMeta) {
putDataFileContent(content, isEnd);
} else {
//其他数据提交方式
}
} else {
//其他请求方式
}
} catch (IOException e) {
throw new RuntimeException(e);
}
}
public Response Response() throws IOException {
Response response = new Response();
byte[] bytes = new byte[1];
ByteArrayOutputStream byteArray = new ByteArrayOutputStream();
int respL = 0;
while (-1 != inputStream.read(bytes)) {
byteArray.write(bytes);
if (bytes[0] == '\n') {
String context = byteArray.toString();
byteArray.reset();
if (context.equals("\r\n")) break;
else if (context.startsWith("Content-Length: "))
respL = Integer.parseInt(context.substring(16).trim());
else if (context.startsWith("Transfer-Encoding: chunked"))
respL = -1;
else if (context.startsWith("HTTP/1.1 "))
response.status = Integer.parseInt(context.substring(9, 12));
}
}
if (-1 == respL) {
ByteArrayOutputStream bodyByte = new ByteArrayOutputStream();
while (-1 != inputStream.read(bytes)) {
byteArray.write(bytes);
if (bytes[0] == '\n') {
int chunked;
String context = byteArray.toString();
chunked = Integer.parseInt(context.trim(), 16);
byteArray.reset();
if (0 == chunked){
response.msg = bodyByte.toString();
break;
}
byte[] chunkedBody = new byte[chunked];
inputStream.read(chunkedBody);
inputStream.read(new byte[2]);
bodyByte.write(chunkedBody);
}
}
} else {
byte[] body = new byte[respL];
inputStream.read(body);
response.msg = new String(body);
}
return response;
}
public void colse() {
try {
outputStream.close();
inputStream.close();
socket.close();
} catch (IOException e) {
throw new RuntimeException(e);
}
}
public void flush() {
try {
if (RequestMothed.POST == requestMothed) {
if (DataMeta.FORM == dataMeta) {
flushForm();
} else {
//其他数据提交方式
}
} else {
//其他请求方式
}
} catch (IOException e) {
throw new RuntimeException(e);
}
}
private void flushForm() throws IOException {
String formM = "----sugar963sugar--";
byte[] bytes = formM.getBytes();
int count = bytes.length;
String countS = Integer.toHexString(count).toLowerCase();
outputStream.write((countS + "\r\n").getBytes());
outputStream.write(bytes);
outputStream.write("\r\n0\r\n\r\n".getBytes());
outputStream.flush();
}
private void putDataOfForm(String key, String value) throws IOException {
String formM = "----sugar963sugar\r\nContent-Disposition: form-data; name=\"" + key + "\"\r\n\r\n" + value + "\r\n";
byte[] bytes = formM.getBytes();
int count = bytes.length;
String countS = Integer.toHexString(count).toLowerCase();
outputStream.write((countS + "\r\n").getBytes());
outputStream.write(bytes);
outputStream.write("\r\n".getBytes());
}
private void putDataOfFormWithFile(String key, String fileName) throws IOException {
String formM = "----sugar963sugar\r\nContent-Disposition: form-data; name=\"" + key + "\";filename=\"" + fileName + "\"\r\nContent-type: text/plain\r\n\r\n";
byte[] bytes = formM.getBytes();
int count = bytes.length;
String countS = Integer.toHexString(count).toLowerCase();
outputStream.write((countS + "\r\n").getBytes());
outputStream.write(bytes);
outputStream.write("\r\n".getBytes());
}
private void putDataFileContent(byte[] content, boolean isEnd) throws IOException {
byte[] contentE = content;
if (isEnd) {
contentE = Arrays.copyOf(content, content.length + 2);
contentE[content.length] = '\r';
contentE[content.length + 1] = '\n';
}
int count = contentE.length;
String countS = Integer.toHexString(count).toLowerCase();
outputStream.write((countS + "\r\n").getBytes());
outputStream.write(contentE);
outputStream.write("\r\n".getBytes());
}
}
利用putDataWithFile和putFileContent可以将文件内容直接写入报文,目前只实现了POST表单提交信息,后期再进行扩充
测试demo:
HttpStream httpStream = new HttpStream(HttpStream.RequestMothed.POST, HttpStream.DataMeta.FORM, "127.0.0.1", 8080, false, "/file/t");
httpStream.putData("api_p","aa22");
httpStream.putDataWithFile("file", "fileName.txt");
httpStream.putFileContent("hah邪恶那大家阿卡\n".getBytes(), false);
httpStream.putFileContent("ddddee".getBytes(), true);
httpStream.putData("f1","0");
httpStream.flush();
HttpStream.Response response = httpStream.Response();
System.out.println(response.status + ":" + response.msg);
httpStream.colse();