Я обновляю реактивный проект (от 0.43.1 -> 0.61.4). Это проект с внутренним бэкэндом, в проекте используется response-native-fileupload (действительно старая библиотека).
Я разбудил это и обновил версию iOS (только что изменил импорт) и сразу же снова заработал. Но версия Android не будет работать, на мой взгляд, код java выглядит нормально, но, очевидно, нет, поскольку он не работает.
Я получаю ошибку 403, без каких-либо сообщений (я У меня нет доступа напрямую к серверу), но поскольку я вижу, что запрос iOS проходит с тем же телом запроса и методом аутентификации. Я не знаю, может ли это быть как-то связано с заголовками или импортом, так как теперь он также работает 64-битный вместо более ранних 32.
Файлы - это изображения (jpeg) и аудио (aa c). И я вижу, как код работает вплоть до ответа без прерывания.
Код раздвоенной библиотеки: https://github.com/PhilippKrone/react-native-fileupload
Android код:
package com.yoloci.fileupload;
import android.os.Bundle;
import com.facebook.react.bridge.Arguments;
import com.facebook.react.bridge.ReadableArray;
import com.facebook.react.bridge.ReadableMap;
import com.facebook.react.bridge.ReadableMapKeySetIterator;
import com.facebook.react.bridge.ReactApplicationContext;
import com.facebook.react.bridge.ReactContextBaseJavaModule;
import com.facebook.react.bridge.ReactMethod;
import com.facebook.react.bridge.Callback;
import java.io.DataInputStream;
import java.net.HttpURLConnection;
import java.net.URL;
import java.io.DataOutputStream;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import com.facebook.react.bridge.WritableMap;
import java.io.FileInputStream;
import org.json.JSONObject;
public class FileUploadModule extends ReactContextBaseJavaModule {
@Override
public String getName() {
return "FileUpload";
}
public FileUploadModule(ReactApplicationContext reactContext) {
super(reactContext);
}
@ReactMethod
public void upload(final ReadableMap options, final Callback callback) {
String lineEnd = "\r\n";
String twoHyphens = "--";
String boundary = "*****";
String uploadUrl = options.getString("uploadUrl");
String method;
if (options.hasKey("method")) {
method = options.getString("method");
} else {
method = "POST";
}
ReadableMap headers = options.getMap("headers");
ReadableArray files = options.getArray("files");
ReadableMap fields = options.getMap("fields");
HttpURLConnection connection = null;
DataOutputStream outputStream = null;
DataInputStream inputStream = null;
URL connectURL = null;
FileInputStream fileInputStream = null;
int bytesRead, bytesAvailable, bufferSize;
byte[] buffer;
int maxBufferSize = 11024;
try {
connectURL = new URL(uploadUrl);
connection = (HttpURLConnection) connectURL.openConnection();
// Allow Inputs & Outputs.
connection.setDoInput(true);
connection.setDoOutput(true);
connection.setUseCaches(false);
connection.setRequestMethod(method);
// set headers
ReadableMapKeySetIterator iterator = headers.keySetIterator();
while (iterator.hasNextKey()) {
String key = iterator.nextKey();
connection.setRequestProperty(key, headers.getString(key));
}
connection.setRequestProperty("Connection", "Keep-Alive");
connection.setRequestProperty("Content-Type", "multipart/form-data;boundary="+boundary);
outputStream = new DataOutputStream( connection.getOutputStream() );
// set fields
ReadableMapKeySetIterator fieldIterator = fields.keySetIterator();
while (fieldIterator.hasNextKey()) {
outputStream.writeBytes(twoHyphens + boundary + lineEnd);
String key = fieldIterator.nextKey();
outputStream.writeBytes("Content-Disposition: form-data; name=\"" + key + "\"" + lineEnd + lineEnd);
outputStream.writeBytes(fields.getString(key));
outputStream.writeBytes(lineEnd);
}
for (int i = 0; i < files.size(); i++) {
ReadableMap file = files.getMap(i);
String filename = file.getString("filename");
String filepath = file.getString("filepath");
filepath = filepath.replace("file://", "");
fileInputStream = new FileInputStream(filepath);
outputStream.writeBytes(twoHyphens + boundary + lineEnd);
outputStream.writeBytes("Content-Disposition: form-data; name=\"image\";filename=\"" + filename + "\"" + lineEnd);
outputStream.writeBytes(lineEnd);
bytesAvailable = fileInputStream.available();
bufferSize = Math.min(bytesAvailable, maxBufferSize);
buffer = new byte[bufferSize];
// Read file
bytesRead = fileInputStream.read(buffer, 0, bufferSize);
while (bytesRead > 0) {
outputStream.write(buffer, 0, bufferSize);
bytesAvailable = fileInputStream.available();
bufferSize = Math.min(bytesAvailable, maxBufferSize);
bytesRead = fileInputStream.read(buffer, 0, bufferSize);
}
outputStream.writeBytes(lineEnd);
}
outputStream.writeBytes(twoHyphens + boundary + twoHyphens + lineEnd);
// Responses from the server (code and message)
int serverResponseCode = connection.getResponseCode();
String serverResponseMessage = connection.getResponseMessage();
if (serverResponseCode != 200) {
fileInputStream.close();
outputStream.flush();
outputStream.close();
callback.invoke("Error happened: " + serverResponseMessage, null);
} else {
BufferedReader br = new BufferedReader(new InputStreamReader(connection.getInputStream()));
StringBuilder sb = new StringBuilder();
String line;
while ((line = br.readLine()) != null) {
sb.append(line);
}
br.close();
String data = sb.toString();
JSONObject mainObject = new JSONObject();
mainObject.put("data", data);
mainObject.put("status", serverResponseCode);
BundleJSONConverter bjc = new BundleJSONConverter();
Bundle bundle = bjc.convertToBundle(mainObject);
WritableMap map = Arguments.fromBundle(bundle);
fileInputStream.close();
outputStream.flush();
outputStream.close();
callback.invoke(null, map);
}
} catch(Exception ex) {
callback.invoke("Error happened: " + ex.getMessage(), null);
}
}
}
Я заметил код:
outputStream.writeBytes("Content-Disposition: form-data; name=\"image\";filename=\"" + filename + "\"" + lineEnd);
, говорящий изображение, но это не имело никакого значения, изменяя его для динамической настройки.
Любой ввод действительно помог бы мне, спасибо заранее!
После тестирования конечных точек на локальном сервере, чтобы увидеть разницу, я получил следующие заголовки: *1024*
Android:
{ accept: 'application/json',
connection: 'Keep-Alive',
'content-type': 'multipart/form-data;boundary=*****',
'user-agent': 'Dalvik/2.1.0 (Linux; U; Android 10; Android SDK built for x86 Build/QSR1.190920.001)',
host: '10.0.2.2:8000',
'accept-encoding': 'gzip',
'content-length': '72936' }
iOS:
{ host: 'localhost:8000',
'content-type': 'multipart/form-data; boundary=----885DEE11-19FF-4A57-B13E-7E0436818ABE',
connection: 'keep-alive',
accept: 'application/json',
'user-agent': 'CFNetwork/1121.2.1 Darwin/18.7.0',
'accept-language': 'en-us',
'accept-encoding': 'gzip, deflate',
'content-length': '202577' }